博弈论(nim游戏,SG函数)
说到自己,就是个笑话。思考问题从不清晰,sg函数的问题证明方法就在眼前可却要弃掉。不过自己理解的也并不透彻,做题也不太行。耳边时不时会想起alf的:“行不行!”
基本的小概念
- 这里我们讨论的是公平游戏(ICG游戏:Impartial Combinatorial Games),满足:
1.双方每步的限制相同(轮流)
2.游戏有尽头 - 对于当前局面的玩家如果能有必胜策略,那就是N局面(反之,P局面)
SG函数
- 每一种决策以及后面的所有可能可以抽象成有向无环图,而sg函数的计算就类似图上dp的过程。
- 若当前的局面是now,下一步的局面为to
\(SG[now]=Mex(SG[to])\)
\(Mex(S)\)表示,最小的没有在\(S\)集合里的非负整数
这就是SG的定义了,这很迷糊,因为它就是一个当前局面的估价,很难说它具体是什么意义(甚至后面还会用它来运算就离谱) - 显然的性质
1.SG[now]后面存在了sg为0的局面,now就为N局面,你接下来你的操作也会move到\(sg[to]=0\)这种局面。
2.1的反之。
3.SG为0表示当前局面P,否则为N。(ps.三个点就是一个结论,3总结1,2) - Muiti-SG的性质(见nim游戏后)
nim游戏
- 这是最基本最重要的游戏,理解它后,你将入门博弈。
- 规则:两个人玩nim游戏,有n堆石子,每堆有\(a_i\)个,每次可从一堆中取走任意多个石子。如果一个人无石子可取就输。
- 结论:\(xorsum\ =\ a_1\ xor\ a_2\ xor\ a_3\ xor\ ...\ xor\ a_n\)是0就为P局面,否则N局面。
- 证明:如果xorsum=0,取后xorsum!=0; 如果xorsum!=0,一定存在从一堆(堆i)取走\(a_i-xorsum\ xor\ a_i\)个石子,使取后\(xorsum=0\)
然后最终输的状态为全0,这是xorsum=0的。而如果一开始为xorsum=0一定被拿捏,一直xorsum=0直到输掉(对面就一直使你xorsum=0) - 这时,你肯定会想:用sg暴力做呢?状态存不下啊。当然就引出了后面有关sg的重要推论……
不过对于只有一堆(或者想成n堆每堆独立)\(sg[i]=i\)的。(i为还剩的石子个数)
重要推论
- 从上面,或许我们能猜想:
S局面被拆分成\(n\)个相互独立的小局面,(我们这里叫S为\(s_1->s_n\)的游戏和,满足:
\(SG[S]\ =\ SG[s_1]\ xor\ SG[s_2]\ ....\ xor\ SG[s_n]\)
这个对于公平游戏来讲,居然是正确的。证明?-> - 证明:
我们通常遇到博弈想最终态(P态为多,因为sg=0)
当然是所有sg值全为0的时候。
你每次只能转移一个小局面(一堆石子),它能转移到0~sg[i]-1以及sg[i]+k,k是不确定的。
你发现如果没有sg[i]+k我们可以用nim游戏来证明。
如果有,相当于加石子,实质上还是符合nim游戏证明中的异或和总是=0和!=0相交替。
加和减因此也没有实质区别,就得证。 - ps.老板告诉我们可以把任何ICG问题转化为相同sg值(=石子个数)的nim游戏。
小结:
nim类,或者ICG类都感觉核心就用到以上几个结论。不过我太菜了,还是做不来题。博弈论还有很多值得探索,不过现实中人也没有那么理性,很多时候也用不到。
有趣题目:
1.字符游戏
- 思路:首先01串+构造长度为L+前缀关系,很容易想到trie树。
因为“互异”,所以每次更新trie树从根开始的一条链,链叶的子树里的点下次就不可以被选择了。
然后我们通过画图找规律,然后每个点的SG值为lowbit(子树深度),画一下就知道了 - 代码:
#include<bits/stdc++.h>
using namespace std;
const int N=1e5+5;
typedef long long ll;
int go[N][2],nd;
ll lowbit(ll u) {return u&(-u);}
ll ans=0;
char s[N];
void Insert() {
int u=0,len=strlen(s);
for(int i=0;i<len;i++) {
int x=s[i]-'0';
if(!go[u][x]) go[u][x]=++nd;
u=go[u][x];
}
}
void dfs(int u,ll dep) {
if(go[u][0]) dfs(go[u][0],dep-1);
else ans^=lowbit(dep);
if(go[u][1]) dfs(go[u][1],dep-1);
else ans^=lowbit(dep);
}
int main() {
int n;ll l;
scanf("%d%lld",&n,&l);
for(int i=1;i<=n;i++) {
scanf("%s",s);
Insert();
}
dfs(0,l);
if(ans)printf("Alice");
else printf("Bob");
return 0;
}
博弈论(nim游戏,SG函数)的更多相关文章
- BZOJ 1874: [BeiJing2009 WinterCamp]取石子游戏 [Nim游戏 SG函数]
小H和小Z正在玩一个取石子游戏. 取石子游戏的规则是这样的,每个人每次可以从一堆石子中取出若干个石子,每次取石子的个数有限制,谁不能取石子时就会输掉游戏. 小H先进行操作,他想问你他是否有必胜策略,如 ...
- POJ 3553 Light Switching Game 博弈论 nim积 sg函数
http://poj.org/problem?id=3533 变成三维的nim积..前面hdu那个算二维nim积的题的函数都不用改,多nim积一次就过了...longlong似乎不必要但是还是加上了 ...
- 组合游戏 - SG函数和SG定理
在介绍SG函数和SG定理之前我们先介绍介绍必胜点与必败点吧. 必胜点和必败点的概念: P点:必败点,换而言之,就是谁处于此位置,则在双方操作正确的情况下必败. N点:必胜点 ...
- 博弈论基础之sg函数与nim
在算法竞赛中,博弈论题目往往是以icg.通俗的说就是两人交替操作,每步都各自合法,合法性与选手无关,只与游戏有关.往往我们需要求解在某一个游戏或几个游戏中的某个状态下,先手或后手谁会胜利的问题.就比如 ...
- 博弈论进阶之SG函数
SG函数 个人理解:SG函数是人们在研究博弈论的道路上迈出的重要一步,它把许多杂乱无章的博弈游戏通过某种规则结合在了一起,使得一类普遍的博弈问题得到了解决. 从SG函数开始,我们不再是单纯的同过找规律 ...
- BZOJ1188 [HNOI2007]分裂游戏(SG函数)
传送门 拿到这道题就知道是典型的博弈论,但是却不知道怎么设计它的SG函数.看了解析一类组合游戏这篇论文之后才知道这道题应该怎么做. 这道题需要奇特的模型转换.即把每一个石子当做一堆石子,且原来在第i堆 ...
- 博弈论初步(SG函数)
讲解见此博客https://blog.csdn.net/strangedbly/article/details/51137432 理解Nim博弈,基于Nim博弈理解SG函数的含义和作用. 学习求解SG ...
- HDU 1536 S-Nim (组合游戏+SG函数)
题意:针对Nim博弈,给定上一个集合,然后下面有 m 个询问,每个询问有 x 堆石子 ,问你每次只能从某一个堆中取出 y 个石子,并且这个 y 必须属于给定的集合,问你先手胜还是负. 析:一个很简单的 ...
- HDU 3032 Nim or not Nim? (sg函数)
Nim or not Nim? Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)T ...
- Wannafly挑战赛23 T2游戏 SG函数
哎,被卡科技了,想了三个小时,最后还是大佬给我说是\(SG\)函数. \(SG\)函数,用起来很简单,证明呢?(不可能的,这辈子都是不可能的) \(SG\)定理 游戏的\(SG\)函数就是各个子游戏的 ...
随机推荐
- vue打包后空白页问题全记录 (background路径,css js404,jsonp等);
总结一下vue打包后问题全记录:大部分开发者webpack基本上都是拿来就用的(并没有系统化的研究). 一 >>> 打包之后的静态文件不能直接访问:(例如dist)打包后搭个服务器才 ...
- 前端面试题整理——手写flatern摊平数组
// flatern 是摊平数组 function flat(arr) { const isDeep = arr.some(item => item instanceof Array) if(! ...
- PAT A1035 Password
题目描述: To prepare for PAT, the judge sometimes has to generate random passwords for the users. The pr ...
- getHibernateTemplate出现的所有find方法的总结
文章转自:http://www.cnblogs.com/DarrenChan/p/5528194.html 一.find(String queryString); 示例:this.getHiberna ...
- JavaScript实现动态表格
运行效果: 源代码: 1 <!DOCTYPE html> 2 <html lang="zh"> 3 <head> 4 <meta char ...
- 在UnityUI中绘制线状统计图
##先来个效果图 觉得不好看可以自己调整 ##1.绘制数据点 线状图一般由数据点和连线组成 在绘制连线之前,我们先标出数据点 这里我选择用Image图片来绘制数据点 新建Canvas,添加空物体Gra ...
- vue3响应式模式设计原理
vue3响应式模式设计原理 为什么要关系vue3的设计原理?了解vue3构建原理,将有助于开发者更快速上手Vue3:同时可以提高Vue调试技能,可以快速定位错误 1.vue3对比vue2 vue2的原 ...
- 1. 初识 JavaScript
1.1 JavaScript 是什么 布兰登·艾奇(Brendan Eich,1961年-). 神奇的大哥用10天完成 JavaScript 设计. 最初命名为 LiveScript,后来在与 Sun ...
- git 将本地文件推送到远程分支的分支
1. 新建文件夹复制远程分支 2. 切换到远程分支 3. 推送到远程 添加到暂存区,先运行 " git add . " 查看文件状态 在运 ...
- 【FAQ】HMS Core广告服务:如何获取正式广告位ID以及流量变现的受限情况
HMS Core广告服务开发指南中提到"xxxx为测试专用的广告位ID,App正式发布时需要改为正式的广告位ID",那么今天咱们就来说说,怎么获取正式的广告位ID. 测试广告位ID ...