试图复制迷宫算法

Posted

技术标签:

【中文标题】试图复制迷宫算法【英文标题】:Trying to replicate a maze algorithm 【发布时间】:2021-04-17 22:06:30 【问题描述】:

我正在尝试使用回溯让我的 C 程序解决迷宫问题。

基本上,程序所做的是将插入的文件作为命令行参数(该文件应包含迷宫结构)并尝试“解决它”,输出所做的每一步。

迷宫墙是hashtags (#),程序应该避开它们。所做的每一步都应标有dot (.)。程序只能踩空格( )

迷宫从符号S (Start)开始,应该到达符号E (End)

迷宫的外观示例如下:

迷宫数据存储在矩阵a[1000][1000]1000 中,在我的程序中定义为MAX。每次我在矩阵a[y][x] 的某个位置上迈出一步,a[y][x] 都应该转换为dot (.)

我正在使用usleep()函数,每80 000 nanoseconds输出一个矩阵,以便用户可以正确看到每一步。

一切正常(几乎没问题)。当迷宫到达某个点而无法继续前进时,它会删除之前的所有步骤,直到找到另一条路为止。问题是,它删除的“点”比它要删除的要多。而且(我认为)这就是它永远不会到达E 点的原因。

代码信息:

这些是我的全局变量声明:

int i_MAX, j_MAX;

int dy[4] = -1, 1, 0, 0;
int dx[4] = 0, 0, -1, 1;

其中i_MAXj_MAX 用于计算迷宫数组a 的大小。

main()函数只包含从命令行参数获取文件的过程,以及计算i_MAXj_MAX,以及调用back(0)

这些是(在我看来)引起麻烦的功能:

int is_acceptable(int step_Y, int step_X, int i, char a[MAX][MAX]) 
    // check if it is possible to move the current point (step_X, step_Y)
    // to the position (step_X+dx[i], step_Y+dy[i])
    step_Y += dy[i];
    step_X += dx[i];
    if (a[step_Y][step_X] == ' ' && step_Y >= 0 && step_X >= 0 
                                 && step_Y < i_MAX && step_X < j_MAX)
        return 1;
    return 0;


int is_solution(int step_Y, int step_X) 
    if (step_Y == i_MAX && step_X == j_MAX) return 1;
    return 0;


void bck(int step_Y, int step_X, char a[MAX][MAX]) 
    // test
    system("clear");
    for (int w = 0; w<i_MAX; w++) 
            for(int x = 0; x<j_MAX; x++)
            putchar(a[w][x]);
        printf("\n");
    
    usleep(80000);
    // end test
    for (int i = 0; i<=3; i++) 
        if (is_acceptable(step_Y, step_X, i, a)) 
            step_Y += dy[i];
            step_X += dx[i];
            a[step_Y][step_X] = '.';
            if (is_solution(step_Y, step_X)) 
                for (int w = 0; w<i_MAX; w++) 
                    for(int x = 0; x<j_MAX; x++)
                        putchar(a[w][x]);
                printf("\n");
                
            
            else bck(step_Y, step_X, a);
            a[step_Y][step_X] = ' ';
        
    

我很确定问题出在bck() 函数的最后一行,在尝试“重置”无效位置时:

a[step_Y][step_X] = ' ';

我尝试以多种方式更改该行,但仍然无法使程序正常运行。

我做错了什么?它与代码的其他部分有关吗?

附:这是我的迷宫文字:

################################################################################
#S                                                                  #          #
################################################################### # ######## #
#                                                           ####### # ##    ## #
# ######################################################### #       # ## ## ## #
# # #E                                                    # ####### # ## ## ## #
# # ##################################################### #         # ## #  ## #
# #                                                       ######### # ## ## ## #
# ####################################################### # #       # ## ## ## #
# #                                                         # ##### # ## ## ## #
# # ####################################################### # # ### # ## ## ## #
# # #        #########################                      # #     # ## ## ## #
# # # ######                         ######################## ##### # ## ## ## #
# # # ## ### ####################### # ##     ###     ##    ##    # # ## ## ## #
# # #  #     #                       #    ###     ###    ##    #### # ## ## ## #
# # ## ############################# # ############################ # ## ## ## #
# # ## #                        ###  #                            # # ## ## ## #
# # ## # ########### ############   ## ############################## ## ## ## #
# #    #           # #            ####                                #  ###   #
# ################ # ####### ######### ##############################   #### # #
#                # # #                                            # ########## #
######## ######### # ############################################## #          #
#                  #                                                  ### ######
################################################################################

【问题讨论】:

尝试找出可能引发问题的最小最简单迷宫。然后使用调试器逐行遍历代码(同时监视变量及其值)以查看实际发生的情况。或者使用大量的调试printf 调用,打印变量值和状态以及其他所有内容。 注意条件a[step_Y][step_X] == ' ' &amp;&amp; step_Y &gt;= 0 &amp;&amp; step_X &gt;= 0 &amp;&amp; step_Y &lt; i_MAX &amp;&amp; step_X &lt; j_MAX。如果step_Ystep_X 为负数(或大于i_MAXj_MAX),您将在检查边界之前超出数组的边界。 @Someprogrammerdude 使用一堆 printf 的唯一垃圾邮件我的控制台日志,我无法查看所有这些 将它们写入文件,或使用调试打印,以便仅在调试控制台中显示。另外,将您的迷宫粘贴为文本,这样我们就不必输入任何内容了。 您可以将调试打印仅添加到代码的一部分,并且一切正常,然后删除它们并在代码的另一部分添加新的打印。或者如@MarkTolonen 所述,将其写入运行后可以通过的文件。 【参考方案1】: 功能:is_solution

E(end) 不一定是 i_MAX 和 j_MAX。在您的示例中,它是 step_Y=5 和 step_X=5

功能:bck

递归问题:在递归调用 bck 之前更改 step_X 和 step_Y。但是,如果这条路是死胡同,在尝试另一个方向之前,您必须将 step_X 和 step_Y 恢复到它们的原始值。

void bck (int step_Y, int step_X, char a[MAX][MAX]) 
    // test
    ...
            else bck(step_Y, step_X, a);
            a[step_Y][step_X] = ' ';
            step_Y -= dy[i];
            step_X -= dx[i];
        
    

【讨论】:

设法修复它。非常感谢!

以上是关于试图复制迷宫算法的主要内容,如果未能解决你的问题,请参考以下文章

算法:深度优先搜索之迷宫寻径

幻想迷宫(或无限迷宫)——因为逻辑不清而做不出的典型

迷宫生成 - 递归除法(它是如何工作的?)

Comet OJ 热身赛(E题)(处理+最短路算法)

以下行完成了啥? [复制]

试图将 cv::Mat 数据复制到 uchar *