zoukankan      html  css  js  c++  java
  • bzoj1085骑士精神(搜索)

    1085: [SCOI2005]骑士精神

    Time Limit: 10 Sec  Memory Limit: 162 MB
    Submit: 1893  Solved: 1051

    Description

      在一个5×5的棋盘上有12个白色的骑士和12个黑色的骑士, 且有一个空位。在任何时候一个骑士都能按照骑
    士的走法(它可以走到和它横坐标相差为1,纵坐标相差为2或者横坐标相差为2,纵坐标相差为1的格子)移动到空
    位上。 给定一个初始的棋盘,怎样才能经过移动变成如下目标棋盘: 为了体现出骑士精神,他们必须以最少的步
    数完成任务。

    Input

      第一行有一个正整数T(T<=10),表示一共有N组数据。接下来有T个5×5的矩阵,0表示白色骑士,1表示黑色骑
    士,*表示空位。两组数据之间没有空行。

    Output

      对于每组数据都输出一行。如果能在15步以内(包括15步)到达目标状态,则输出步数,否则输出-1。

    Sample Input

    2
    10110
    01*11
    10111
    01001
    00000
    01011
    110*1
    01110
    01010
    00100

    Sample Output

    7
    -1
    /*
    迭代加深dfs经典题!
    记录目标状态,然后从起始状态搜索。
    爆搜可能超时,要加剪枝 
    */
    #include<cstdio>
    #include<cstring>
    #include<iostream>
    #define limit 15
    using namespace std;
    int T,ans=20,flag;
    int xx[9]={0,-2,-2,-1,-1,1,1,2,2};
    int yy[9]={0,-1,1,-2,2,2,-2,1,-1};
    int s[10][7],get[10][10];
    int target[10][10];
    char si;
    
    void get_t()//记录目标状态 
    {
        target[1][1]=1;target[1][2]=1;target[1][3]=1;target[1][4]=1;target[1][5]=1;
        target[2][1]=0;target[2][2]=1;target[2][3]=1;target[2][4]=1;target[2][5]=1;
        target[3][1]=0;target[3][2]=0;target[3][3]=2;target[3][4]=1;target[3][5]=1;
        target[4][1]=0;target[4][2]=0;target[4][3]=0;target[4][4]=0;target[4][5]=1;
        target[5][1]=0;target[5][2]=0;target[5][3]=0;target[5][4]=0;target[5][5]=0;
    }
    
    int Judge()//计算当前状态与目标状态至少还有多少步 
    {
        int ret=0;
        for(int i=1;i<=5;i++)
          for(int j=1;j<=5;j++)
          {
              if(s[i][j]!=target[i][j])
                ret++;
          }
        return ret;
    }
    
    void DFS(int now,int x,int y,int sum)
    {
        if(flag) return;
        int c=Judge();
        if(now==sum)
        {
            if(c==0)
              flag=1,ans=sum;
        }
        if(now-1+c>sum) return;//最优性剪枝:当前的步数+差异>限制步数 
        for(int i=1;i<=8;i++)
        {
            int nx=x+xx[i];
            int ny=y+yy[i];
            if(nx>0&&nx<=5&&ny>0&&ny<=5)
            {
                swap(s[x][y],s[nx][ny]);
                DFS(now+1,nx,ny,sum);
                swap(s[x][y],s[nx][ny]);
            }
        }
    }
    
    int main()
    {
        scanf("%d",&T);
        get_t();
        while(T--)
        {
            int x,y;
            for(int i=1;i<=5;i++)
              for(int j=1;j<=5;j++)
              {
                  cin>>si;
                  if(si=='*'){x=i;y=j;get[i][j]=2;}    
                else get[i][j]=si-'0';
              }
            for(int k=0;k<=limit;k++)//迭代加深限制步数 
            {
                flag=0;ans=20;
                for(int i=1;i<=5;i++)
                  for(int j=1;j<=5;j++)
                    s[i][j]=get[i][j];
                DFS(0,x,y,k);
                if(ans==k) break;
            }
            if(ans<=15)printf("%d
    ",ans);
            else printf("-1
    ");
        }
        return 0;
    } 
    心若向阳,无谓悲伤
    折花枝,恨花枝,准拟花开人共卮,开时人去时。 怕相思,已相思,轮到相思没处辞,眉间露一丝。
  • 相关阅读:
    mysql增量同步到greenplum
    c笔记06--编译与作用域
    C笔记05-选择顺序结构,关系与相等,优先级和结合性
    C笔记02-C数据类型与数据类型转换
    C笔记01-C简介与补码
    jQuery属性操作之.val()函数
    jQuery属性操作之.attr()
    jQuery笔记: 基本概念与jQuery核心
    笔记: js构造函数与原型
    布尔运算符
  • 原文地址:https://www.cnblogs.com/L-Memory/p/6159812.html
Copyright © 2011-2022 走看看