title: CodeForces 786A Berzerk

data: 2018-3-3 10:29:40

tags:

  • 博弈论
  • bfs
  • 无限
  • with draw

    copyright: true

    categories:
  • 信息学竞赛
  • 题目

    description: 有一个环, 1的位置是黑洞, 有一个怪物在任意位置上, 两个人依次行动, 可以将怪物移动自己决策集合中的任意步, 谁先把它送进黑洞谁就赢, 请输出当怪物在任意位置和谁先手的各自的胜负情况, 有平局.

A. Berzerk

description

  Rick and Morty are playing their own version of Berzerk (which has nothing in common with the famous Berzerk game). This game needs a huge space, so they play it with a computer.

  In this game there are n objects numbered from 1 to n arranged in a circle (in clockwise order). Object number 1 is a black hole and the others are planets. There's a monster in one of the planet. Rick and Morty don't know on which one yet, only that he's not initially in the black hole, but Unity will inform them before the game starts. But for now, they want to be prepared for every possible scenario.

  Each one of them has a set of numbers between 1 and n - 1 (inclusive). Rick's set is s1 with k1 elements and Morty's is s2 with k2 elements. One of them goes first and the player changes alternatively. In each player's turn, he should choose an arbitrary number like x from his set and the monster will move to his x-th next object from its current position (clockwise). If after his move the monster gets to the black hole he wins.

  Your task is that for each of monster's initial positions and who plays first determine if the starter wins, loses, or the game will stuck in an infinite loop. In case when player can lose or make game infinity, it more profitable to choose infinity game.

Input

  The first line of input contains a single integer n (\(2 ≤ n ≤ 7000\)) — number of objects in game.

  The second line contains integer k1 followed by k1 distinct integers s1, 1, s1, 2, ..., s1, k1 — Rick's set.

  The third line contains integer k2 followed by k2 distinct integers s2, 1, s2, 2, ..., s2, k2 — Morty's set

\(1 ≤ ki ≤ n - 1\ and\ 1 ≤ si, 1, si, 2, ..., si, ki ≤ n - 1\ for\ 1 ≤ i ≤ 2.\)

Output

  In the first line print n - 1 words separated by spaces where i-th word is "Win" (without quotations) if in the scenario that Rick plays first and monster is initially in object number i + 1 he wins, "Lose" if he loses and "Loop" if the game will never end.

  Similarly, in the second line print n - 1 words separated by spaces where i-th word is "Win" (without quotations) if in the scenario that Morty plays first and monster is initially in object number i + 1 he wins, "Lose" if he loses and "Loop" if the game will never end.

Examples

input #1

5

2 3 2

3 1 2 3

output #1

Lose Win Win Loop

Loop Win Win Win

input #2

8

4 6 2 3 4

2 3 6

output #2

Win Win Win Win Win Win Win

Lose Win Lose Lose Win Lose Lose

Way:

  这显然是一个无限博弈问题, 因为两人都选择最佳策略, 所以两个人有可能一直僵持下去谁也奈何不了谁, 这显然就是平局.

  显然可以用搜索搜出每个状态的所有后继状态, 根据必败态和必胜态的定义.

  • 终止状态是必败态.
  • 有一个后继状态是必败态的状态是必胜态.
  • 所以后继状态都是必胜态的状态是必败态.

  通过这样可以推出所有的必胜态和必败态.可是平局如何定义呢?后继状态有不确定状态的状态是不确定状态, 可以发现它构成了一个环.

Code

#include<algorithm>
#include<iostream>
#include<cstdlib>
#include<cstdio>
#include<queue>
#define N 7005
using namespace std; namespace Input{
inline int read(){
char ch=getchar();int s=0;
for(;!isdigit(ch);ch=getchar());
for(;isdigit(ch);s=s*10+ch-'0',ch=getchar());
return s;
}
inline void read(int &s){
char ch=getchar();
for(;!isdigit(ch);ch=getchar());
for(s=0;isdigit(ch);s=s*10+ch-'0',ch=getchar());
}
};using namespace Input;
struct Move{
int pos;
bool now;
Move(int _pos=0,bool _now=1){pos=_pos;now=_now;}
}; int mark[N][2],alt[N][2];
int vis[N][2],du[N][2];
int f[N][2],s[2],n; int mov(int pos,int step){
return (pos-step+n)%n;
}
void bfs(){
queue<Move>que;
f[0][1]=f[0][0]=2;
que.push((Move){0,0});
que.push((Move){0,1});
vis[0][1]=true;
vis[0][0]=true;
while(!que.empty()){
Move top=que.front();que.pop();
bool opt=top.now^1;
for(int i=0;i<s[opt];++i){
int to=mov(top.pos,alt[i][opt]);
if(f[top.pos][top.now]==1){
mark[to][opt]++;
if(!vis[to][opt]&&mark[to][opt]==s[opt]){
f[to][opt]=2;
vis[to][opt]=true;
que.push((Move){to,opt});
}
}
else if(!vis[to][opt]){
f[to][opt]=1;
vis[to][opt]=true;
que.push((Move){to,opt});
}
}
}
} int main(){
read(n);
read(s[0]);
for(int i=0;i<s[0];++i)read(alt[i][0]);
read(s[1]);
for(int i=0;i<s[1];++i)read(alt[i][1]);
bfs();int ans;
for(int i=1;i<n;++i){
ans=f[i][0];
printf(ans?ans==1?"Win":"Lose":"Loop");
putchar(' ');
}
putchar('\n');
for(int i=1;i<n;++i){
ans=f[i][1];
printf(ans?ans==1?"Win":"Lose":"Loop");
putchar(' ');
}
system("pause");
return 0;
}

cf786a的更多相关文章

  1. CF786A - Berzerk

    /* CF786A - Berzerk http://codeforces.com/contest/786/problem/A 博弈论 直接搜出NP状态图.记得要记忆化剪枝. * */ #includ ...

随机推荐

  1. Android View 绘制刷新流程分析

    Android中对View的更新有很多种方式,使用时要区分不同的应用场合.1.不使用多线程和双缓冲      这种情况最简单,一般只是希望在View发生改变时对UI进行重绘.你只需显式地调用View对 ...

  2. 【BZOJ4520】K远点对(KD-Tree)

    [BZOJ4520]K远点对(KD-Tree) 题面 BZOJ 洛谷 题解 考虑暴力. 维护一个大小为\(K\)的小根堆,然后每次把两个点之间的距离插进去,然后弹出堆顶 这样子可以用\(KD-Tree ...

  3. 无序数组中第Kth大的数

    题目:找出无序数组中第Kth大的数,如{63,45,33,21},第2大的数45. 输入: 第一行输入无序数组,第二行输入K值. 该是内推滴滴打车时(2017.8.26)的第二题,也是<剑指of ...

  4. bzoj1257: [CQOI2007]余数之和sum(数论)

    非常经典的题目... 要求 则有 实际上 最多只有2*sqrt(k)种取值,非常好证明 因为>=sqrt(k)的数除k下取整得到的数一定<=sqrt(k),而k除以<=sqrt(k) ...

  5. 一些实用的JQuery代码片段收集

    本文将展示50个非常实用的JQuery代码片段,这些代码能够给你的JavaScript项目提供帮助.其中的一些代码段是从jQuery1.4.2才开始支持的做法,另一些则是真正有用的函数或方法,他们能够 ...

  6. linux内存管理及手动释放机制

    inux系统中查看内存状态一般都会用到free linux的free命令中,cached和buffers的区别 Free Mem:表示物理内存统计 -/+ buffers/cached:表示物理内存的 ...

  7. Ubuntu16.04+Cuda8.0+1080ti+caffe+免OpenCV3.2.0+faster-rCNN教程

    一.事先声明:1.Ubuntu版本:Ubuntu使用的是16.04.而不是16.04.1或16.04.2,这三个是有区别的.笔者曾有过这样的经历,Git上一个SLAM地图构建程序在Ubuntu14.0 ...

  8. Codeforces Round #341 (Div. 2)B

    B. Wet Shark and Bishops time limit per test 2 seconds memory limit per test 256 megabytes input sta ...

  9. Android数据存储与访问

    1.文件    1)保存到手要内存,文件保存到/data/data对应的应用程序包下面    如    FILE_PATH = "/data/data/com.diysoul.filedem ...

  10. linux内核挂载文件系统的两种方式

    1.nfs方式(挂载的为文件夹) bootargs=root=/dev/nfs nfsroot=192.168.1.105:/mnt/rootfs/rootfs ip=192.168.1.88:192 ...