洛谷4363 [九省联考2018]一双木棋chess

2023-11-29 04:40

本文主要是介绍洛谷4363 [九省联考2018]一双木棋chess,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

标签:状压,记忆化搜索

题目

题目传送门

题目描述

菲菲和牛牛在一块n 行m 列的棋盘上下棋,菲菲执黑棋先手,牛牛执白棋后手。
棋局开始时,棋盘上没有任何棋子,两人轮流在格子上落子,直到填满棋盘时结束。

落子的规则是:一个格子可以落子当且仅当这个格子内没有棋子且这个格子的左侧及上方的所有格子内都有棋子。

棋盘的每个格子上,都写有两个非负整数,从上到下第i 行中从左到右第j 列的格
子上的两个整数记作 Ai,j A i , j Bi,j B i , j 。在游戏结束后,菲菲和牛牛会分别计算自己的得分:菲菲的得分是所有有黑棋的格子上的 Ai,j A i , j 之和,牛牛的得分是所有有白棋的格子上的 Bi,j B i , j 的和。

菲菲和牛牛都希望,自己的得分减去对方的得分得到的结果最大。现在他们想知道,在给定的棋盘上,如果双方都采用最优策略且知道对方会采用最优策略,那么,最终的结果如何。

输入输出格式

输入格式

从文件chess.in 中读入数据。

输入第一行包含两个正整数n;m,保证n;m <= 10。

接下来n 行,每行m 个非负整数,按从上到下从左到右的顺序描述每个格子上的
第一个非负整数:其中第i 行中第j 个数表示 Ai,j A i , j

接下来n 行,每行m 个非负整数,按从上到下从左到右的顺序描述每个格子上的
第二个非负整数:其中第i 行中第j 个数表示 Bi,j B i , j

输出格式

输出到文件chess.out 中。

输出一个整数,表示菲菲的得分减去牛牛的得分的结果。

输入输出样例

输入样例#1

2 3
2 7 3
9 1 2
3 7 2
2 3 1

输出样例#1

2

说明

样例1说明

棋盘如图所示,双方都采用最优策略时,棋局如下:

• 菲菲下在第1 行第1 列(这是第一步时唯一可以落子的格子);

• 牛牛下在第1 行第2 列;

• 菲菲下在第2 行第1 列;

• 牛牛下在第1 行第3 列;

• 菲菲下在第2 行第2 列;

• 牛牛下在第2 行第3 列(这是这一步时唯一可以落子的格子);

• 填满棋盘,游戏结束,盘面如下。

菲菲的得分为:2 + 9 + 1 = 12 ;牛牛的得分为:7 + 2 + 1 = 10 。

对于所有的测试数据,n;m <= 10 , Ai,j A i , j ; Bi,j B i , j <= 100000。

对于编号为奇数的测试点,保证所有的 Bi,j B i , j = 0 。

分析

幸亏这不是自己AH的省选,这题在考场上我肯定A不掉(果然菜鸡啊)


这东西好像叫什么对抗搜索???雾

因为A想最大化 ab ∑ a − ∑ b 而B想要最小化 ab ∑ a − ∑ b

考虑到每行放的棋子数量肯定是递减的

. . . . .
. . . .
. . . 
. .

这样的形式,而且肯定是从左向右的

我们用一个 m+1 m + 1 进制 n n <script type="math/tex" id="MathJax-Element-87">n</script>位的数来储存每行放旗子的状态

那么就可以用记忆化来优化这个搜索

用map来存储结果即可


话说我用洛谷的题号是不是政治很正确啊(逃)

code

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cmath>
#include<cstring>
#include<algorithm>
#include<map>
#define rep(i,a,b) for(int i=a;i<=b;i++)
#define dep(i,a,b) for(int i=a;i>=b;i--)
#define ll long long
#define mem(x,num) memset(x,num,sizeof x)
#define reg(x) for(int i=last[x];i;i=e[i].next)
using namespace std;
inline ll read(){ll f=1,x=0;char ch=getchar();while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}while(ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();}return x*f;
}
//**********head by yjjr**********
#define inf 1e9
const int maxn=26;
ll End;int n,m,num[maxn],a[maxn][maxn],b[maxn][maxn];
map<ll,int> mp;
inline int unzip(ll sta){int s=0;dep(i,n,1)s+=(num[i]=(sta%(m+1))),sta/=(m+1);return s&1;
}//判断当前该谁下棋
inline ll zip(){ll s=0;rep(i,1,n)s=s*(m+1)+num[i];return s;
}//压缩状态
inline int dfs(ll sta){if(mp.find(sta)!=mp.end())return mp[sta];if(sta==End)return 0;//如果搜索到最终状态,那么就退出int opt=unzip(sta),ans=opt?inf:-inf;if(num[1]<m){++num[1];if(opt)ans=min(ans,dfs(zip())-b[1][num[1]]);else ans=max(ans,dfs(zip())+a[1][num[1]]);num[1]--;}rep(i,2,n)if(num[i-1]>num[i]){++num[i];if(opt)ans=min(ans,dfs(zip())-b[i][num[i]]);else ans=max(ans,dfs(zip())+a[i][num[i]]);num[i]--;}return mp[sta]=ans;
}
int main(){n=read(),m=read();rep(i,1,n)rep(j,1,m)a[i][j]=read();rep(i,1,n)rep(j,1,m)b[i][j]=read();rep(i,1,n)num[i]=m;End=zip();dfs(0);cout<<mp[0]<<endl;return 0;
}

这篇关于洛谷4363 [九省联考2018]一双木棋chess的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/431478

相关文章

BUUCTF靶场[web][极客大挑战 2019]Http、[HCTF 2018]admin

目录   [web][极客大挑战 2019]Http 考点:Referer协议、UA协议、X-Forwarded-For协议 [web][HCTF 2018]admin 考点:弱密码字典爆破 四种方法:   [web][极客大挑战 2019]Http 考点:Referer协议、UA协议、X-Forwarded-For协议 访问环境 老规矩,我们先查看源代码

2018秋招C/C++面试题总结

博主从8月中旬开始大大小小面试了十几家公司,至今也许是告一段落吧,希望后面会有好结果,因此总结记录一些C/C++方向常见的问题。和大家一起学习! 参考了互联网的各种资源,自己尝试归类整理,谢谢~ 一、C和C++的区别是什么? C是面向过程的语言,C++是在C语言的基础上开发的一种面向对象编程语言,应用广泛。 C中函数不能进行重载,C++函数可以重载 C++在C的基础上增添类,C是一个结构

大厂算法例题解之网易2018秋招笔试真题 (未完)

1、字符串碎片 【题目描述】一个由小写字母组成的字符串可以看成一些同一字母的最大碎片组成的。例如,“aaabbaaac” 是由下面碎片组成的:‘aaa’,‘bb’,‘c’。牛牛现在给定一个字符串,请你帮助计算这个字符串的所有碎片的 平均长度是多少。 输入描述: 输入包括一个字符串 s,字符串 s 的长度 length(1 ≤ length ≤ 50),s 只含小写字母(‘a’-‘z’) 输出描述

高精度计算(代码加解析,洛谷p1601,p1303)除法待更新

目录 高精度加法 高精度减法 高精度乘法 高精度加法 我们知道在c++语言中任何数据类型都有一定的表示范围。当两个被加数很大时,正常加法不能得到精确解。在小学,我们做加法都采用竖式方法。那么我们也只需要按照加法进位的方式就能得到最终解。 8 5 6+ 2 5 5-------1 1 1 1 加法进位: c[i] = a[i] + b[i];if(c[i] >=

洛谷 凸多边形划分

T282062 凸多边形的划分 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 先整一个半成品,高精度过两天复习一下补上 #include <iostream>#include <algorithm>#include <set>#include <cstring>#include <string>#include <vector>#include <map>

能量项链,洛谷

解释:  环形dp问题还是考虑将环拉直,可以参考我上一篇文章:环形石子合并 [2 3 5 10 2] 3 5 10 将环拉直,[]内是一个有效的区间,可以模拟吸收珠子的过程,         如[2 3 5] <=>(2,3)(3,5)    2是头,3是中间,5是尾 len >= 3:因为最后[2 10 2]是最小的可以合并的有效区间 len <= n + 1因为[2 3

vulhub GhostScript 沙箱绕过(CVE-2018-16509)

1.执行以下命令启动靶场环境并在浏览器访问 cd vulhub/ghostscript/CVE-2018-16509 #进入漏洞环境所在目录   docker-compose up -d #启动靶场   docker ps #查看容器信息 2.访问网页 3.下载包含payload的png文件 vulhub/ghostscript/CVE-2018-16509/poc.png at

Python JAVA接口UTC 时间 '2018-08-06T10:00:00.000Z' 格式转化为本地时间

Python JAVA接口UTC 时间 '2018-08-06T10:00:00.000Z' 格式转化为本地时间 方法1 import datetimeorigin_date_str= "2019-07-26T08:20:54Z"utc_date = datetime.datetime.strptime(origin_date_str, "%Y-%m-%dT%H:%M:%SZ")loca

最值求解 | 管理类联考数学专项

日期内容2024.9.5新建2024.9.6曦曦求最值完结 实数求最值至少至多抽屉原理工程问题线性规划一次性绝对值求最值 参考: b站跟着曦曦老师玩转【最值】

洛谷P5490扫描线

0是最小的数字,将一个线段看成一个区间,对于一个矩形,从下扫到上,入边为1,而出边为-1,意思是将这个区间上的所有点加1(区间修改).把线段表示为Line[i],其中记录了l,r,h,tag,左右端点,高度,入边还是出边(1或-1) 那么每次区间修改后不为0的区间它的值可能是1,2,3或者是其它数字,这不好统计,可以将它转化一下,0是不是表示没有被覆盖过的地方,我们只要统计0的个数然后用总长减去