测试地址:The Rotation Game

题目大意:一个井字形的棋盘上有8个1,8个2,8个3(图的话测试地址里面有),有8个操作(标号为A~H),表示把某一行或某一列做旋转变换,例如A方向所对的列:A <- 1 1 1 2 2 2 3 3,进行A操作后就是:A <- 1 1 2 2 2 3 3 1。要使最中心的八个方块的数字相同,求出字典序最小的操作序列(如果无需操作,输入“No moves needed”)和最后中心八个方块上的数字。

做法:迭代加深搜索,从0开始增大搜索上限,然后就可以DFS了。

然而,直接裸着写会TLE,那么怎么办呢?可以做可行性剪枝。我们知道每做一次操作最多只能减少1个和其他不同的数字,也就是说,如果中心八个方块出现最多的数字个数是n,那么就至少需要(8-n)次才能转变为目标状态,所以,如果当前步数加上(8-n)大于搜索上限,那么就可以剪掉。

还有一个小小的优化,不知道有没有效果,就是记录一个操作的反操作(就是操作的是同一行(或列)而方向相反的操作),并记录上一步的操作,如果当前枚举到的是上一步操作的反操作,那么就没有意义(相当于又移回上上步的状态了),可以剪掉。

然后,再简单处理一下旋转操作就可以过了。

以下是本人代码:

#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <iostream>
#include <algorithm>
#define inf 999999999
using namespace std;
int a[30],ans,route[310];
int move[8][7]=
{
  {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}
},back[9]={5,4,7,6,1,0,3,2,8},checklist[8]={7,8,9,12,13,16,17,18};

int read()
{
  int i=1;
  while(i<=24&&scanf("%d",&a[i])!=EOF) i++;
  return i-1;
}

bool check()
{
  bool flag=1;
  for(int i=0;i<=7;i++)
    if (a[checklist[i]]!=a[7]) {flag=0;break;}
  return flag;
}

void swapper(int dir)
{
  int temp=a[move[dir][0]];
  for(int i=1;i<=6;i++)
    a[move[dir][i-1]]=a[move[dir][i]];
  a[move[dir][6]]=temp;
}

bool f(int step)
{
  int stat[4]={0},m=0;
  for(int i=0;i<=7;i++)
  {
    stat[a[checklist[i]]]++;
    if (stat[a[checklist[i]]]>m) m=stat[a[checklist[i]]];
  }
  return step+8-m<=ans;
}

bool dfs(int last,int step)
{
  if (step==ans)
  {
    if (check())
	{
	  if (step==0) {printf("No moves needed\n");}
	  else
	  {
	    for(int i=1;i<=step;i++)
	      printf("%c",route[i]);
		printf("\n");
	  }
	  return 1;
	}
	else return 0;
  }
  for(int i=0;i<8;i++)
    if (i!=back[last])
	{
	  swapper(i);route[step+1]='A'+i;
	  if (f(step)&&dfs(i,step+1)) return 1;
	  swapper(back[i]);
	}
  return 0;
}

int main()
{
  while(read()!=1)
  {
    for(ans=0;ans<=inf;ans++)
	  if (dfs(8,0)) break;
	printf("%d\n",a[7]);
  }
  
  return 0;
}