luogu2346 四子连棋
题目大意
在一个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 四子连棋的更多相关文章
- codevs1004四子连棋[BFS 哈希]
1004 四子连棋 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 黄金 Gold 题目描述 Description 在一个4*4的棋盘上摆放了14颗棋子,其中有7颗 ...
- Codevs p1004 四子连棋
四子连棋 题目描述 Description 在一个4*4的棋盘上摆放了14颗棋子,其中有7颗白色棋子,7颗黑色棋子,有两个空白地带,任何一颗黑白棋子都可以向 ...
- 【宽度优先搜索】神奇的状态压缩 CodeVs1004四子连棋
一.写在前面 其实这是一道大水题,而且还出在了数据最水的OJ上,所以实际上这题并没有什么难度.博主写这篇blog主要是想写下一个想法--状态压缩.状态压缩在记录.修改状态以及判重去重等方面有着极高的( ...
- codevs 1004 四子连棋
1004 四子连棋 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 黄金 Gold 题目描述 Description 在一个4*4的棋盘上摆放了14颗棋子,其中有7颗白 ...
- codevs 1004 四子连棋 BFS、hash判重
004 四子连棋 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 黄金 Gold 题目描述 Description 在一个4*4的棋盘上摆放了14颗棋子,其中有7颗白色棋 ...
- 【洛谷 P2346】四子连棋(状态压缩,搜索)
其实这题可以直接二进制状压做,1表示黑棋,0表示白棋,另外记录下2个空点的位置就行了. 具体看代码(冗长): #include <iostream> #include <cstdio ...
- 迭代加深搜索[codevs1004 四子连棋]
迭代加深搜索 一.算法简介 迭代加深搜索是在速度上接近广度优先搜索,空间上和深度优先搜索相当的搜索方式.由于在使用过程中引入了深度优先搜索,所以也可以当作深度优先搜索的优化方案. 迭代加深搜索适用于当 ...
- codevs1004四子连棋
1004 四子连棋 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 黄金 Gold 题目描述 Description 在一个4*4的棋盘上摆放了14颗棋子,其中有7颗白 ...
- P2346 四子连棋
P2346 四子连棋 迭代加深++ 题意描述 在一个4*4的棋盘上摆放了14颗棋子,其中有7颗白色棋子,7颗黑色棋子,有两个空白地带,任何一颗黑白棋子都可以向上下左右四个方向移动到相邻的空格,这叫行棋 ...
随机推荐
- GC策略
JVM里的GC(Garbage Collection)的算法有很多种,如标记清除收集器,压缩收集器,分代收集器等等,详见HotSpot VM GC 的种类 现在比较常用的是分代收集(ge ...
- CMU Database Systems - Two-phase Locking
首先锁是用来做互斥的,解决并发执行时的数据不一致问题 如图会导致,不可重复读 如果这里用lock就可以解决,数据库里面有个LockManager来作为master,负责锁的记录和授权 数据库里面的基本 ...
- Python之UDP编程
参考原文 廖雪峰Python教程 TCP是建立可靠连接,并且通信双方都可以以流的形式发送数据.相对TCP,UDP则是面向无连接的协议. 使用UDP协议时,不需要建立连接,只需要知道对方的IP地址和端口 ...
- python_ 学习笔记(基本数据类型)
python3有6中标准数据类型:Number(数字).String(字符串).List(列表).Tuple(元组).Dictionary(字典).Set(集合)不可变数据:Number.String ...
- java一维数组的声明、初始化及排序
public class TestArray { public static void main(String[] args) { /** 数组声明及动态初始化 int a[] = new int[a ...
- 腾讯云:iptables基础
iptables 基础 iptables 基本命令 任务时间:5min ~ 10min iptables 可以简单理解为 Linux 系统内核级防火墙 netfilter 的用户态客户端. Linux ...
- BZOJ 1012 洛谷1198 最大数 maxnumber
用线段数维护即可 #include<cstdio> #include<algorithm> #define ls (cur<<1) #define rs (cur& ...
- WeChat-小程序-tabbar
WeChat-小程序-tabbar https://developers.weixin.qq.com/miniprogram/dev/framework/config.html#%E5%85%A8%E ...
- java使用JNA框架调用dll动态库
这两天了解了一下java调用dll动态库的方法,总的有三种:JNI.JNA.JNative.其中JNA调用DLL是最方便的. ·JNI ·JNA ·JNative java使用 JNI来调用dll动态 ...
- Mzc家中的男家丁
题目背景 mzc与djn的…还没有众人皆知,所以我们要来宣传一下. 题目描述 mzc家很有钱(开玩笑),他家有n个男家丁,现在mzc要将她们全都聚集起来(干什么就不知道了).现在知道mzc与男家丁们互 ...