题目大意

  在一个4*4的棋盘上摆放了14颗棋子,其中有7颗白色棋子,7颗黑色棋子,有两个空白地带,任何一颗黑白棋子都可以向上下左右四个方向移动到相邻的空格,这叫行棋一步,黑白双方交替走棋,任意一方可以先走,如果某个时刻使得任意一种颜色的棋子形成四个一线(包括斜线),这样的状态为目标棋局。求用最少的步数移动到目标棋局的步数。

  总体思路很简单,Bfs即可,只是需要注意以下几点:

  • memcmp的返回值不一定是-1, 0, 1,而是<0, =0, >0的某个数。这在windows和linux上的效果不一样。
  • 注意:黑白双方交替走棋。
  • 任意一方都必须走一步。
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <queue>
#include <set>
#include <cassert>
using namespace std; const int MAX_N = 10;
const int N = 4;
const int Dir[4][2] = { {1, 0}, {0, 1}, {-1, 0}, {0, -1} }; struct Node
{
char A[MAX_N][MAX_N];
int Level;
char NextColor; Node()
{
memset(A, 0, sizeof(A));
Level = 0;
} Node operator = (const Node& a)
{
memcpy(A, a.A, sizeof(A));
Level = a.Level;
NextColor = a.NextColor;
return *this;
} bool operator < (const Node& a) const
{
if (NextColor != a.NextColor)
return NextColor == 'B';
else
return memcmp(A, a.A, sizeof(A)) < 0;
} bool operator == (const Node& a) const
{
return NextColor == a.NextColor && memcmp(A, a.A, sizeof(A)) == 0;
} void OPos1(int &oRow1, int &oCol1)
{
for (int row = 1; row <= N; row++)
for (int col = 1; col <= N; col++)
if (A[row][col] == 'O')
{
oRow1 = row;
oCol1 = col;
return;
}
} void OPos2(int &oRow2, int &oCol2)
{
int oRow1, oCol1;
OPos1(oRow1, oCol1);
for (int col = oCol1 + 1; col <= N; col++)
if (A[oRow1][col] == 'O')
{
oRow2 = oRow1;
oCol2 = col;
return;
}
for (int row = oRow1 + 1; row <= N; row++)
for (int col = 1; col <= N; col++)
if (A[row][col] == 'O')
{
oRow2 = row;
oCol2 = col;
return;
}
assert(0);
} bool CanMove1(const int dRow, const int dCol)
{
int oRow1, oCol1;
OPos1(oRow1, oCol1);
int nextRow = oRow1 + dRow, nextCol = oCol1 + dCol;
return A[nextRow][nextCol] == NextColor && nextRow <= N && nextRow >= 1 && nextCol <= N && nextCol >= 1;
} Node GetMove1(int dRow, int dCol)
{
int oRow1, oCol1;
OPos1(oRow1, oCol1);
Node ans = *this;
swap(ans.A[oRow1][oCol1], ans.A[oRow1 + dRow][oCol1 + dCol]);
return ans;
} bool CanMove2(const int dRow, const int dCol)
{
int oRow2, oCol2;
OPos2(oRow2, oCol2);
int nextRow = oRow2 + dRow, nextCol = oCol2 + dCol;
return A[nextRow][nextCol] == NextColor && nextRow <= N && nextRow >= 1 && nextCol <= N && nextCol >= 1;
} Node GetMove2(int dRow, int dCol)
{
int oRow2, oCol2;
OPos2(oRow2, oCol2);
Node ans = *this;
swap(ans.A[oRow2][oCol2], ans.A[oRow2 + dRow][oCol2 + dCol]);
return ans;
} bool Ok()
{
for (int row = 1; row <= N; row++)
{
char st = A[row][1];
bool ok = true;
for (int col = 2; col <= N; col++)
if (A[row][col] != st)
{
ok = false;
break;
}
if (ok)
return true;
}
for (int col = 1; col <= N; col++)
{
char st = A[1][col];
bool ok = true;
for (int row = 2; row <= N; row++)
if (A[row][col] != st)
{
ok = false;
break;
}
if (ok)
return true;
}
char st = A[1][1];
bool ok = true;
for (int i = 2; i <= N; i++)
if (A[i][i] != st)
{
ok = false;
break;
}
if (ok)
return true;
st = A[1][N];
ok = true;
for (int row = 2, col = N - 1; row <= N; row++, col--)
if (A[row][col] != st)
{
ok = false;
break;
}
return ok;
}
};
Node Start; int Bfs()
{
static queue<Node> q;
static set<Node> cache;
Node s1 = Start, s2 = Start;
s1.NextColor = 'B';
s2.NextColor = 'W';
q.push(s1);
q.push(s2);
cache.insert(s1);
cache.insert(s2);
while (!q.empty())
{
Node cur = q.front();
q.pop();
if (!(cur == s1 || cur == s2) && cur.Ok())
return cur.Level;
for (int i = 0; i < 4; i++)
{
if (cur.CanMove1(Dir[i][0], Dir[i][1]))
{
Node next = cur.GetMove1(Dir[i][0], Dir[i][1]);
next.NextColor = (cur.NextColor == 'B' ? 'W' : 'B');
if (!cache.count(next))
{
next.Level = cur.Level + 1;
cache.insert(next);
q.push(next);
}
}
if (cur.CanMove2(Dir[i][0], Dir[i][1]))
{
Node next = cur.GetMove2(Dir[i][0], Dir[i][1]);
next.NextColor = (cur.NextColor == 'B' ? 'W' : 'B');
if (!cache.count(next))
{
next.Level = cur.Level + 1;
cache.insert(next);
q.push(next);
}
}
}
}
return -1;
} int main()
{
for (int i = 1; i <= 4; i++)
scanf("%s", Start.A[i] + 1);
printf("%d\n", Bfs());
return 0;
}

  

luogu2346 四子连棋的更多相关文章

  1. codevs1004四子连棋[BFS 哈希]

    1004 四子连棋   时间限制: 1 s   空间限制: 128000 KB   题目等级 : 黄金 Gold   题目描述 Description 在一个4*4的棋盘上摆放了14颗棋子,其中有7颗 ...

  2. Codevs p1004 四子连棋

                          四子连棋 题目描述 Description 在一个4*4的棋盘上摆放了14颗棋子,其中有7颗白色棋子,7颗黑色棋子,有两个空白地带,任何一颗黑白棋子都可以向 ...

  3. 【宽度优先搜索】神奇的状态压缩 CodeVs1004四子连棋

    一.写在前面 其实这是一道大水题,而且还出在了数据最水的OJ上,所以实际上这题并没有什么难度.博主写这篇blog主要是想写下一个想法--状态压缩.状态压缩在记录.修改状态以及判重去重等方面有着极高的( ...

  4. codevs 1004 四子连棋

    1004 四子连棋  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 黄金 Gold     题目描述 Description 在一个4*4的棋盘上摆放了14颗棋子,其中有7颗白 ...

  5. codevs 1004 四子连棋 BFS、hash判重

    004 四子连棋 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 黄金 Gold       题目描述 Description 在一个4*4的棋盘上摆放了14颗棋子,其中有7颗白色棋 ...

  6. 【洛谷 P2346】四子连棋(状态压缩,搜索)

    其实这题可以直接二进制状压做,1表示黑棋,0表示白棋,另外记录下2个空点的位置就行了. 具体看代码(冗长): #include <iostream> #include <cstdio ...

  7. 迭代加深搜索[codevs1004 四子连棋]

    迭代加深搜索 一.算法简介 迭代加深搜索是在速度上接近广度优先搜索,空间上和深度优先搜索相当的搜索方式.由于在使用过程中引入了深度优先搜索,所以也可以当作深度优先搜索的优化方案. 迭代加深搜索适用于当 ...

  8. codevs1004四子连棋

    1004 四子连棋  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 黄金 Gold     题目描述 Description 在一个4*4的棋盘上摆放了14颗棋子,其中有7颗白 ...

  9. P2346 四子连棋

    P2346 四子连棋 迭代加深++ 题意描述 在一个4*4的棋盘上摆放了14颗棋子,其中有7颗白色棋子,7颗黑色棋子,有两个空白地带,任何一颗黑白棋子都可以向上下左右四个方向移动到相邻的空格,这叫行棋 ...

随机推荐

  1. 【sqli-labs】 对于less34 less36的宽字节注入的一点深入

    1.AddSlashes() 首先来观察一下是如何通过构造吃掉转义字符的 先将less 34的网页编码换成gbk 加上一些输出 echo "Before addslashes(): &quo ...

  2. webpack常见问题

    概念问题一:什么是webpack和grunt和gulp有什么不同 答案:Webpack是一个模块打包器,他可以递归的打包项目中的所有模块,最终生成几个打包后的文件.他和其他的工具最大的不同在于他支持c ...

  3. SpringBoot项目编译后没有xxxmapper.xml文件解决方法

    在pom.xml文件中添加如下代码 <build> <plugins> <plugin> <groupId>org.springframework.bo ...

  4. python3.x Day2 购物车程序练习

    购物车程序: 1.启动程序后,输入用户名密码后,如果是第一次登录,让用户输入工资,然后打印商品列表 2.允许用户根据商品编号购买商品 3.用户选择商品后,检测余额是否够,够就直接扣款,不够就提醒 4. ...

  5. 文件内容差异对比-python

    上次没精力时候,看了下python自动化运维,给print加了颜色,新鲜哒 今天来写写文件对比 step1:引入difflib库(无需安装,python自带) step2:将文件内容按行分割,spli ...

  6. 教你如何使用Python写游戏辅助脚本

    主要实现方式是通过图片的对比,在游戏中就行点击.运行程序需要以下东西. PIL: 图片处理模块     (python3 换成了 pillow)  下载地址: https://www.lfd.uci. ...

  7. STM32 配置PC13~PC15

    在STM32的数据手册的管脚分配图中可以看到:PC14与OSC32_IN公用一个引脚,PC15与OSC32_OUT公用一个引脚,它们的使用方法如下: 当LSE(低速外部时钟信号)开启时,这两个公用管脚 ...

  8. [linux]centos7下解决yum install mysql-server没有可用包

    第一步:安装从网上下载文件的wget命令 [root@master ~]# yum -y install wget 第二步:下载mysql的repo源 [root@master ~]# wget ht ...

  9. 如何相互转换逗号分隔的字符串和List --https://blog.csdn.net/yywusuoweile/article/details/50315377

    如何相互转换逗号分隔的字符串和List ---https://blog.csdn.net/yywusuoweile/article/details/50315377 方法 2: 利用Guava的Joi ...

  10. hdu 1533KM算法

    #include<stdio.h> #include<string.h> #include<math.h> #define inf 0x3fffffff #defi ...