Codeforces 786A Berzerk(博弈论)
?
【題目鏈接】?http://codeforces.com/problemset/problem/786/A
?
【題目大意】
有兩個(gè)人,每個(gè)人有一個(gè)數(shù)集,里面有一些數(shù),現(xiàn)在有一個(gè)環(huán),有個(gè)棋子放在1,
有個(gè)不確定位置的終點(diǎn),兩個(gè)人輪流從自己的數(shù)集中選擇一個(gè)數(shù),作為這個(gè)棋子移動(dòng)的步數(shù)
問終點(diǎn)在不同位置,不同人先手的時(shí)候誰能贏,或者游戲陷入循環(huán)
?
【題解】
我們從st_0_0=st_1_0=0開始倒著推導(dǎo),
如果一個(gè)狀態(tài)是必?cái)B(tài),那么它的前繼節(jié)點(diǎn)一定是必勝態(tài)
如果一個(gè)點(diǎn)的所有后繼都是必勝態(tài),那么這個(gè)節(jié)點(diǎn)一定是必?cái)B(tài)。
每當(dāng)一個(gè)點(diǎn)被其必勝后繼推導(dǎo)到,那么其度數(shù)減一,當(dāng)度數(shù)為0時(shí)則表示其為必?cái)B(tài)
我們根據(jù)這些結(jié)論倒著推導(dǎo)每個(gè)狀態(tài)的答案并記錄,最后按順序輸出即可。
?
?
【代碼】
#include <cstdio> #include <vector> #include <cstring> #include <string> using namespace std; const int N=100010; int n,sg[2][N],d[2][N],k,x; vector<int> g[2]; int dfs(int k,int pos,int v){int &ret=sg[k][pos];if(~ret)return ret;ret=v;if(v==0){for(int i=0;i<g[k^1].size();i++){int x=g[k^1][i];int j=(pos+n-x)%n;if(j==0)continue;dfs(k^1,j,1);}}else{for(int i=0;i<g[k^1].size();i++){int x=g[k^1][i];int j=(pos+n-x)%n;if(j==0)continue;if(--d[k^1][j]==0)dfs(k^1,j,0);}}return ret; } int main(){scanf("%d",&n);for(int i=0;i<2;i++){scanf("%d",&k);g[i].clear();while(k--){scanf("%d",&x);g[i].push_back(x);}for(int j=1;j<n;j++)d[i][j]=g[i].size();}memset(sg,-1,sizeof(sg));dfs(0,0,0);dfs(1,0,0);string s[3]={"Loop","Lose","Win"};for(int k=0;k<2;k++){for(int i=1;i<n;i++){printf("%s%c",s[sg[k][i]+1].c_str(),i+1==n?'\n':' ');}}return 0; }轉(zhuǎn)載于:https://www.cnblogs.com/forever97/p/codeforces786a.html
總結(jié)
以上是生活随笔為你收集整理的Codeforces 786A Berzerk(博弈论)的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: label标签中的for属性
- 下一篇: 怎么退出自适应巡航_你的自适应巡航“全速