【BZOJ3106】[CQOI2013] 棋盘游戏(对抗搜索)
大致题意: 在一张\(n*n\)的棋盘上有一枚黑棋子和一枚白棋子。白棋子先移动,然后是黑棋子。白棋子每次可以向上下左右四个方向中任一方向移动一步,黑棋子每次则可以向上下左右四个方向中任一方向移动一至二步。当某游戏者把自己的棋子移动到对方棋子所在的格子时,他就赢了。两个游戏者都很聪明,可以获胜时会尽快获胜,必输时会尽量拖延时间。试判断谁会赢,需要多少回合。
对抗搜索
这道题的做法应该是对抗搜索。
一波简单的分析
我们先来对题目进行一波简单的分析。
不难发现,因为黑棋每次能走的步数大于白棋每次能走的步数,所以除非白棋第一步就吃掉黑棋,否则白棋必输。
既然这样,我们只需特判白棋获胜的情况,然后题目就转换成了求黑棋追上白棋所需的时间。
这样一来,就变成了一道较简单的对抗搜索题了。
直接用记忆化优化即可(当然,理论上来讲\(Alpha-Beta\)剪枝也可以做,但我没去试过)。
代码
#include<bits/stdc++.h>
#define max(x,y) ((x)>(y)?(x):(y))
#define min(x,y) ((x)<(y)?(x):(y))
#define abs(x) ((x)<0?-(x):(x))
#define LL long long
#define ull unsigned long long
#define swap(x,y) (x^=y,y^=x,x^=y)
#define Fsize 100000
#define tc() (FinNow==FinEnd&&(FinEnd=(FinNow=Fin)+fread(Fin,1,Fsize,stdin),FinNow==FinEnd)?EOF:*FinNow++)
#define pc(ch) (putchar(ch))
#define N 20
int OutputTop=0;char Fin[Fsize],*FinNow=Fin,*FinEnd=Fin,OutputStack[Fsize];
using namespace std;
int n,X1,Y1,X2,Y2,res[N+1][N+1][N+1][N+1][2][3*N+1];
inline void read(int &x)
{
x=0;static char ch;
while(!isdigit(ch=tc()));
while(x=(x<<3)+(x<<1)+ch-48,isdigit(ch=tc()));
}
inline void write(int x)
{
if(!x) return (void)pc('0');
while(x) OutputStack[++OutputTop]=x%10+48,x/=10;
while(OutputTop) pc(OutputStack[OutputTop]),--OutputTop;
}
inline int dfs(int X1,int Y1,int X2,int Y2,int Which,int Step)//对抗搜索
{
if(Step>3*n) return 1e9;//深度限制,以防无限制地搜索下去(可以保证最终的答案≤3n)
if(res[X1][Y1][X2][Y2][Which][Step]) return res[X1][Y1][X2][Y2][Which][Step];//如果已经访问当前状态,就返回上次求解出的答案
if(X1==X2&&Y1==Y2) return Which?1e9:0;//如果已经重合了,就退出函数
register int i,t,ans=Which?1e9:0;
Which^=1,++Step;//更新Which和Step为下一个状态,以避免不断地运算
if(Which)//如果下一个轮黑棋操作,即当前为白棋操作
{
if(X1>1) t=dfs(X1-1,Y1,X2,Y2,Which,Step),ans=max(ans,t);
if(X1<n) t=dfs(X1+1,Y1,X2,Y2,Which,Step),ans=max(ans,t);
if(Y1>1) t=dfs(X1,Y1-1,X2,Y2,Which,Step),ans=max(ans,t);
if(Y1<n) t=dfs(X1,Y1+1,X2,Y2,Which,Step),ans=max(ans,t);
}
else//如果当前为黑棋操作
{
if(X2>1) t=dfs(X1,Y1,X2-1,Y2,Which,Step),ans=min(ans,t);
if(X2>2) t=dfs(X1,Y1,X2-2,Y2,Which,Step),ans=min(ans,t);
if(X2<=n-1) t=dfs(X1,Y1,X2+1,Y2,Which,Step),ans=min(ans,t);
if(X2<=n-2) t=dfs(X1,Y1,X2+2,Y2,Which,Step),ans=min(ans,t);
if(Y2>1) t=dfs(X1,Y1,X2,Y2-1,Which,Step),ans=min(ans,t);
if(Y2>2) t=dfs(X1,Y1,X2,Y2-2,Which,Step),ans=min(ans,t);
if(Y2<=n-1) t=dfs(X1,Y1,X2,Y2+1,Which,Step),ans=min(ans,t);
if(Y2<=n-2) t=dfs(X1,Y1,X2,Y2+2,Which,Step),ans=min(ans,t);
}
return res[X1][Y1][X2][Y2][Which^1][Step-1]=ans+1;
}
int main()
{
read(n),read(X1),read(Y1),read(X2),read(Y2);
if(abs(X1-X2)+abs(Y1-Y2)<=1) return puts("WHITE 1"),0;//特判白棋获胜的情况
return pc('B'),pc('L'),pc('A'),pc('C'),pc('K'),pc(' '),write(dfs(X1,Y1,X2,Y2,0,1)),0;
}
【BZOJ3106】[CQOI2013] 棋盘游戏(对抗搜索)的更多相关文章
- [bzoj3106][cqoi2013][棋盘游戏] (对抗搜索+博弈论)
Description 一个n*n(n>=2)棋盘上有黑白棋子各一枚.游戏者A和B轮流移动棋子,A先走. l A的移动规则:只能移动白棋子.可以往上下左右四个方向之一移动一格. ...
- bzoj3106 [cqoi2013]棋盘游戏
Description 一个n*n(n>=2)棋盘上有黑白棋子各一枚.游戏者A和B轮流移动棋子,A先走. l A的移动规则:只能移动白棋子.可以往上下左右四个方向之一移动一格. ...
- bzoj千题计划200:bzoj3106: [cqoi2013]棋盘游戏
http://www.lydsy.com/JudgeOnline/problem.php?id=3106 白棋如果第一步不能赢,那么一定输 因为可以黑棋走的距离比白棋大,黑棋可以下一步吃掉白棋,也可以 ...
- BZOJ 3106: [cqoi2013]棋盘游戏(对抗搜索)
题目:http://www.lydsy.com/JudgeOnline/problem.php?id=3106 对抗搜索,f[x][y][a][b][c][d]表示当前谁走,走了几步,及位置. (因为 ...
- 【BZOJ 3106】 3106: [cqoi2013]棋盘游戏 (对抗搜索)
3106: [cqoi2013]棋盘游戏 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 544 Solved: 233 Description 一个 ...
- 博弈论经典算法(一)——对抗搜索与Alpha-Beta剪枝
前言 在一些复杂的博弈论题目中,每一轮操作都可能有许多决策,于是就会形成一棵庞大的博弈树. 而有一些博弈论题没有什么规律,针对这样的问题,我们就需要用一些十分玄学的算法. 例如对抗搜索. 对抗搜索简介 ...
- 3106: [cqoi2013]棋盘游戏
3106: [cqoi2013]棋盘游戏 链接 分析: 极大极小搜索 + 记忆化. 代码 #include<bits/stdc++.h> using namespace std; type ...
- BZOJ.5248.[九省联考2018]一双木棋chess(对抗搜索 记忆化)
BZOJ 洛谷P4363 [Update] 19.2.9 重做了遍,感觉之前写的有点扯= = 首先棋子的放置情况是阶梯状的. 其次,无论已经放棋子的格子上哪些是黑棋子哪些是白棋子,之前得分如何,两人在 ...
- P2962 [USACO09NOV]灯Lights 对抗搜索
\(\color{#0066ff}{题目描述}\) 贝希和她的闺密们在她们的牛棚中玩游戏.但是天不从人愿,突然,牛棚的电源跳闸了,所有的灯都被关闭了.贝希是一个很胆小的女生,在伸手不见拇指的无尽的黑暗 ...
随机推荐
- go语言实战教程:实战项目资源导入和项目框架搭建
从本节内容开始,我们将利用我们所学习的Iris框架的相关知识,进行实战项目开发. 实战项目框架搭建 我们的实战项目是使用Iris框架开发一个关于本地服务平台的后台管理平台.平台中可以管理用户.商品.商 ...
- hdu6070(分数规划/二分+线段树区间更新,区间最值)
题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=6070 题意: 给出一个题目提交序列, 从中选出一个正确率最小的子串. 选中的子串中每个题目当且仅当最 ...
- bzoj 3123: [Sdoi2013]森林(45分暴力)
3123: [Sdoi2013]森林 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 4184 Solved: 1235[Submit][Status ...
- C语言的头文件和宏定义详解
原文链接:https://blog.csdn.net/abc_12366/article/details/79155540
- JD孔_20160901
1.买的 “[京东超市]GL格朗 耳温枪/电子体温计/温度计/耳温计EW-2” http://item.jd.com/385507.html 2.
- poj(2406) kmp
题目链接:https://vjudge.net/problem/POJ-2406 kmp学习:https://blog.csdn.net/starstar1992/article/details/54 ...
- Codeforces Beta Round #79 (Div. 1 Only) B. Buses 树状数组
http://codeforces.com/contest/101/problem/B 给定一个数n,起点是0 终点是n,有m两车,每辆车是从s开去t的,我们只能从[s,s+1,s+2....t-1 ...
- (转)运维老鸟教你安装centos6.5如何选择安装包
运维老鸟教你安装centos6.5如何选择安装包 原文:http://blog.51cto.com/oldboy/1564620 近来发现越来越多的运维小伙伴们都有最小化安装系统的洁癖,因此,找老男孩 ...
- ZK数据同步
整个集群完成Learner选举后,Learner向Leader发起注册,然后进入数据同步环节,即Leader将learner上缺失的事务请求同步给Learner 1). 获取Learner状态 根据L ...
- C#面试常见题
1. 简述 private. protected. public. internal 修饰符的访问权限. 答: private : 私有成员, 在类的内部才可以访问. protected : 保护成员 ...