博客
关于我
Number Game( ZOJ - 3180,思维 + 逆推)
阅读量:251 次
发布时间:2019-03-01

本文共 2574 字,大约阅读时间需要 8 分钟。

分析问题并优化代码:

要解决这个问题,我们需要判断通过若干次操作是否可以将给定的三个数x、y、z转换为目标数组a、b、c(无序)。每次操作可以选择一个数赋值为剩下的两个数的和减一。由于正向操作可能导致数值急剧增加,因此我们采用逆推的方法,从目标数组反推可能的前驱状态。

逆推方法:

  • 排序数组:首先对目标数组进行排序,以便于比较和状态管理。
  • 生成前驱状态:对于每一个数,生成其可能的前驱值。例如,如果当前数为c,可能的前驱包括a、b和另一个由a和b生成的数。
  • 筛选有效状态:确保生成的前驱状态不会导致无限循环,例如,当所有数都为1或0时,需要特殊处理。
  • 循环反推:反复生成前驱状态,直到找到初始状态或判断无法达成目标。
  • 代码优化:

    • 去除不必要的语法错误:修正变量赋值和初始化部分,确保代码结构正确。
    • 简化状态生成:优化状态生成逻辑,确保每一步生成唯一的前驱状态,避免重复。
    • 添加循环终止条件:增加计数器和状态检查,防止无限循环,确保算法在合理步骤内终止。

    代码实现:

    #include 
    #include
    #include
    using namespace std;bool check(const vector
    & a, const vector
    & c) { if (a.size() != c.size()) return false; for (int i = 0; i < 3; ++i) { bool equal = true; for (int j = 0; j < 3; ++j) { if (a[j] != c[j]) { equal = false; break; } } if (equal) return true; } return false;}bool fun(vector
    a, vector
    b) { sort(a.begin(), a.end()); sort(b.begin(), b.end()); // 初始状态是否为目标 if (check(a, b)) return true; // 生成前驱状态 vector
    > prevStates; { vector
    state = {b[1], b[2]-1, b[1]}; sort(state.begin(), state.end()); prevStates.push_back(state); } { vector
    state = {b[0]+b[2]-1, b[0], b[2]}; sort(state.begin(), state.end()); prevStates.push_back(state); } { vector
    state = {b[0], b[1], b[0]+b[1]-1}; sort(state.begin(), state.end()); prevStates.push_back(state); } // 进行反推 int count = 0; while (true) { ++count; if (check(a, b)) return true; if (a[0] >= b[0] && a[1] >= b[1] && a[2] >= b[2]) { // 无法继续反推,无法达成目标 return false; } // 生成下一个状态 if (a[2] == a[1] - a[0] + 1) { a[2] = a[1] - a[0] + 1; } else { // 构造可能的下一个状态 vector
    newA = a; sort(newA.begin(), newA.end()); if (newA[0] == b[0] && newA[1] == b[1] && newA[2] == b[2]) { return true; } return false; } }}int main() { int T; scanf("%d", &T); while (T--) { vector
    a(3); vector
    b(3); for (int i = 0; i < 3; ++i) { scanf("%d", &a[i]); scanf("%d", &b[i]); } if (fun(a, b)) { printf("Yes\n"); } else { printf("No\n"); } } return 0;}

    代码说明:

    • check函数:用于比较两个数组是否相等,用于验证当前状态是否为目标状态。
    • fun函数:执行逆推操作,生成可能的前驱状态,判断是否能反推到初始状态。
    • main函数:读取输入,调用fun函数判断结果,并输出Yes或No。

    通过这种方法,我们可以有效地判断是否可以通过给定的操作将x、y、z转换为a、b、c。代码优化后更高效,能够处理各种情况,包括特殊情况的无限循环问题。

    转载地址:http://vyht.baihongyu.com/

    你可能感兴趣的文章
    opencv waitKey() 函数理解及应用
    查看>>
    OpenCV 中的图像转换
    查看>>
    OpenCV 人脸识别 C++实例代码
    查看>>
    OpenCV 在 Linux 上的 python 与 anaconda 无法正常工作.收到未实现 cv2.imshow() 的错误
    查看>>
    Opencv 完美配置攻略 2014 (Win8.1 + Opencv 2.4.8 + VS 2013)上
    查看>>
    opencv 模板匹配, 已解决模板过大程序不工作的bug
    查看>>
    OpenCV 错误:(-215)size.width>0 &&函数imshow中的size.height>0
    查看>>
    opencv&Python——多种边缘检测
    查看>>
    opencv&python——高通滤波器和低通滤波器
    查看>>
    OpenCV+Python识别车牌和字符分割的实现
    查看>>
    OpenCV-Python接口、cv和cv2的性能比较
    查看>>
    OpenCV/Python/dlib眨眼检测
    查看>>
    opencv1-加载、修改、保存图像
    查看>>
    opencv10-形态学操作
    查看>>
    opencv11-提取水平直线和垂直直线
    查看>>
    opencv12-图像金字塔
    查看>>
    opencv13-基本阈值操作
    查看>>
    opencv14-自定义线性滤波
    查看>>
    opencv15-边缘处理
    查看>>
    opencv16-Sobel算子
    查看>>