推箱子游戏-用matlab找到解

xiaoxiao2021-02-28  47

一个简单的手机推箱子游戏图,自己没找到答案,只好编程求解了。复制代码后,以文件名move_box.m存盘,即可在matlab中执行。每次按下回车键,就会移动一次箱子。修改文件中的grds,可求解其他格局的问题。工人路径没有优化,所以有重复移动的现象。用C++实现的话,只需要5秒钟可求解。

function [x]=move_box()set(0,'RecursionLimit',1000);clc;global tar grds MAX_BOXES MAX_STEPS his total_steps;% 箱子的目标位置,用于判断是否成功% 每一行是一个箱子的下标tar = [4,3;4,4;4,6;6,6];% 历史数据,避免再进入同一格局his = containers.Map;% 总步数,用于显示进度total_steps = 0;%{把每次移动得到的结果称为一个格局。问题的初始格局和目标格局分别为 o o o o o o o o o o o o o o o o o o . X . o o o o o . . . o o o o o . o B . . o o o . o . . . o o . B . . . . o o . B B . B . o o . . B B . o o o . . . . . o o o o o . o . o o o o o . o B o o o o o . . . o o o o o . . . o o o o o o o o o o o o o o o o o o约定 B -- 箱子 o -- 墙壁 X -- 工人 . -- 空白%}% 最大步数MAX_STEPS = 1000;% 箱子个数MAX_BOXES = 4;% 格局grds(8,8,MAX_STEPS)=0;grds(:,:,1)=[    'oooooooo';    'oo.X.ooo';    'oo.oB..o';    'o.B....o';    'o..BB.oo';    'ooo.o.oo';    'ooo...oo';    'oooooooo'];% 计算工人位置[x,y]=find(grds(:,:,1)=='X');% 求解s = test(1,x,y);if s > 0    for i = 1:s        print_grd(i);        fprintf(1,'\n');        pause;    endelse    fprintf(1, 'failed\n');endend% 判断是否成功function [res] = is_succ(step)global MAX_BOXES tar grds;% 检查每一个目标位置是否都有箱子res = true;for i = 1:MAX_BOXES    if grds(tar(i,1),tar(i,2),step) ~= 'B'        res = false;        break;    end;end;end% 打印格局function print_grd(step)global grds;for i=1:8    for j=1:8        fprintf(1,'%c ',char(grds(i,j,step)));    end    fprintf(1,'\n');endend% 判断是否可以沿方向d移动function [res] = can_move(d, step, x, y)global grds;% d的取值为1到4的整数,分别表示上下左右四个方向switch d    % 向上移动    case 1        % 上方是墙壁吗?        if grds(x-1,y,step) == 'o'            % 返回不能移动            res = false;            % 上方是空白吗?        elseif grds(x-1,y,step) == '.'            % 返回可以移动            res = true;            % 上方既不是墙壁、也不是空白,那么只能是箱子        else            % 如果箱子上方是空白,则能移动,否则不能移动            res = grds(x-2,y,step) == '.';        end;    case 2        if grds(x+1,y,step) == 'o'            res = false;        elseif grds(x+1,y,step) == '.'            res = true;        else            res = grds(x+2,y,step) == '.';        end    case 3        if grds(x,y-1,step) == 'o'            res = false;        elseif grds(x,y-1,step) == '.'            res = true;        else            res = grds(x,y-2,step) == '.';        end;    case 4        if grds(x,y+1,step) == 'o'            res = false;        elseif grds(x,y+1,step) == '.'            res = true;        else            res = grds(x,y+2,step) == '.';        endendend% 按方向d计算step+1步的新格局function move(d, step, x, y)global grds;% 新格局位于grds[step + 1]处,大多数元素与grds[step]相同grds(:,:,step+1) = grds(:,:,step);switch (d)    case 1        % 工人上方是箱子        if grds(x-1,y,step+1)=='B'            % 箱子上移            grds(x-2,y,step+1) = 'B';        end;        % 工人上移        grds(x-1,y,step+1) = 'X';    case 2        if grds(x+1,y,step+1) == 'B'            grds(x+2,y,step+1) = 'B';        end        grds(x+1,y,step+1) = 'X';    case 3        if grds(x,y-1,step+1) == 'B'            grds(x,y-2,step+1) = 'B';        end        grds(x,y-1,step+1) = 'X';    case 4        if grds(x,y+1,step+1) == 'B'            grds(x,y+2,step+1) = 'B';        end        grds(x,y+1,step+1) = 'X';end% 工人原来的位置变为空白grds(x,y,step+1) = '.';end% 把格局变为字符串function [res]=get_key(step, x, y)global grds;x = ['0'+x,',','0'+y];for i = 2:7    for j = 2:7        if grds(i,j,step) == 'B'            x = [x,',','0'+i,',','0'+j];        end    endendres = char(x);end% 是否是老格局,如果不是,在his中记录function [res]=is_dup(step, x, y)global his;k = get_key(step,x,y);try    % 如果是新格局会抛出异常    his(k);    res = true;catch    res = false;    % 在his中记录新格局,1并无实际意义    his(k) = 1;end;end% 搜索,处理第step步移动function [res] = test(step, x, y)global MAX_STEPS total_steps;res = step;total_steps = total_steps+1;if mod(total_steps,1000)==0    fprintf(1,'total_steps = %d\n', total_steps);end% 超出搜索范围,做失败处理if (step > MAX_STEPS)    fprintf(1,'too deep\n');    res = 0;    return;end;% 成功if is_succ(step)    return;end% 沿上下左右四个方向搜索一步for d = 1:4    % 判断是否可以沿方向d移动    if can_move(d, step, x, y) == false        continue;    end;    % 执行沿方向d的移动    move(d, step, x, y);    x1 = x;    y1 = y;    switch d        case 1            x1 = x - 1;        case 2            x1 = x + 1;        case 3            y1 = y - 1;        case 4            y1 = y + 1;    end    % 判断移动后是否为重复格局    if is_dup(step+1, x1, y1)        continue;    end    res = test(step+1,x1,y1);    if res > 0        return;    endendres = 0;end

转载请注明原文地址: https://www.6miu.com/read-2621259.html

最新回复(0)