The Rotation Game (POJ 2286) 题解

【问题描述】

     (由于是英文的,看不懂,这里就把大意给大家说一下吧……都是中国人,相信大家也不愿意看英文……)

      图一

       如图,一个井字形的棋盘,中间有着1-3任意的数,有ABCDEFGH八个操作,每个操作意味着该操作所在行朝该操作方向整体移动一格,详见图,你的目的是:对于输入的多组数据,用最少的步数使得井字形棋盘中间的八个数为同一个数,若不需操作就已达到要求,则输出“No moves needed”,无论是否需要操作,你都应将中间的数字给输出。输入以一个0结束。

【样例输入】

    1 1 1 1 3 2 3 2 3 1 3 2 2 3 1 2 2 2 3 1 2 1 3 3

    1 1 1 1 1 1 1 1 2 2 2 2 2 2 2 2 3 3 3 3 3 3 3 3

    0

【样例输出】  

    AC

    2

    DDHH

    2

【解题思路】

     又是一道求最少步数的题目……果断写广搜,然而,写完广搜后,我奇迹般地发现……MLE了…………………………空间只给了150MB,我再一看时间………………15000ms……也不想多说什么了,赶紧用上今天刚学的IDA*算法。

    IDA*算法是一种将估价函数与深搜结合的搜索方法,搜索方式是用深度优先搜索的方式,搜完第k层若还没有结果,那么就退出,然后从第一层开始搜到第k+1层,不断加深所搜的层数,因此又叫迭代加深搜索。其实说这么复杂,按我的理解,就是用深搜来做的广搜……

    现在我们来看一下题目。因为是井字型棋盘,且输入还是一行数,那么,我们就给棋盘上的数标上号,分别为1-24,那么八个移动的过程也可以表示出来了,然后由于是深搜,那么就还有回溯的过程,对于每一个操作,都需要一个反操作,而题目中正好又提示了我们,A的反操作是F等等,因此,我们只需要记录每个操作的反操作是第几个就行了。

    然后是边界条件,这里的边界条件是当搜索层数大于当前所设置的最大深度限制时,便退出,实际上每一个迭代加深搜索都有这个边界条件,而另外的边界条件则因题目而异,这道题则不需要其他的边界条件,满足要求退出即可。

    接下来考虑初始化的问题,我们在找中间的值的时候,自然是要找最多的数,然后将其他的数移成这个数就行了,那么估价函数就是8-最大的数的个数,从这个估价函数的层数开始搜索,详见代码。

【代码实现】

 1 const xh:array[1..8,1..7] of longint=((1,3,7,12,16,21,23),(2,4,9,13,18,22,24),(11,10,9,8,7,6,5),(20,19,18,17,16,15,14),(24,22,18,13,9,4,2),(23,21,16,12,7,3,1),(14,15,16,17,18,19,20),(5,6,7,8,9,10,11));
 2       fan:array[1..8] of longint=(6,5,8,7,2,1,4,3);
 3       op:array[1..8] of char=('A','B','C','D','E','F','G','H');
 4       mid:array[1..8] of longint=(7,8,9,12,13,16,17,18);
 5 var n,m,dep:longint;
 6     s:array[1..24] of longint;
 7     a:array[1..50] of longint;
 8     i,j:longint;
 9 function max(a,b,c:longint):longint;
10 begin
11  max:=a;
12  if b>max then
13   max:=b;
14  if c>max then
15   max:=c;
16 end;
17 function get:longint;
18 var i:longint;
19     cnt:array[1..4] of longint;
20 begin
21  fillchar(cnt,sizeof(cnt),0);
22  for i:=1 to 8 do
23   inc(cnt[s[mid[i]]]);
24  exit(8-max(cnt[1],cnt[2],cnt[3]));
25 end;
26 procedure move(k:longint);
27 var i,t:longint;
28 begin
29  t:=s[xh[k,1]];
30  for i:=1 to 6 do
31   s[xh[k,i]]:=s[xh[k,i+1]];
32  s[xh[k,7]]:=t;
33 end;
34 function dfs(k:longint):boolean;
35 var i,h:longint;
36 begin
37  if k>=dep then
38   exit(false);
39  for i:=1 to 8 do
40   begin
41    move(i);//移动
42    a[k]:=i;
43    h:=get;//求最多的数的个数
44    if h=0 then exit(true);
45    if (k+h<dep)and(dfs(k+1)) then//深搜
46     exit(true);
47    move(fan[i]);//回溯
48   end;
49  exit(false);
50 end;
51 begin
52  read(s[1]);
53  while s[1]<>0 do
54   begin
55    for i:=2 to 24 do
56     read(s[i]);
57    dep:=get;
58    if dep=0 then
59     begin
60      writeln('No moves needed');//所给数据本就满足要求,输出,退出
61      writeln(s[7]);
62      read(s[1]);
63      continue;
64     end;
65    while not(dfs(1)) do//如果不满足要求,加深层数,再进行深搜
66     inc(dep);
67    for i:=1 to dep-1 do
68     write(op[a[i]]);
69    writeln;
70    writeln(s[7]);
71    read(s[1]);
72   end;
73 end.

 

转载于:https://www.cnblogs.com/PengBoLiuXu/p/4513610.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值