• 尼姆博弈


    尼姆博奕(Nimm Game):有三堆各若干个物品,两个人轮流从某一堆取任意多的
    物品,规定每次至少取一个,多者不限,最后取光者得胜。

        这种情况最有意思,它与二进制有密切关系,我们用(a,b,c)表示某种局势,首
    先(0,0,0)显然是奇异局势,无论谁面对奇异局势,都必然失败。第二种奇异局势是
    (0,n,n),只要与对手拿走一样多的物品,最后都将导致(0,0,0)。仔细分析一
    下,(1,2,3)也是奇异局势,无论对手如何拿,接下来都可以变为(0,n,n)的情
    形。

        计算机算法里面有一种叫做按位模2加,也叫做异或的运算,我们用符号(+)表示
    这种运算。这种运算和一般加法不同的一点是1+1=0。先看(1,2,3)的按位模2加的结
    果:

    1 =二进制01
    2 =二进制10
    3 =二进制11 (+)
    ———————
    0 =二进制00 (注意不进位)

        对于奇异局势(0,n,n)也一样,结果也是0。

        任何奇异局势(a,b,c)都有a(+)b(+)c =0。

    如果我们面对的是一个非奇异局势(a,b,c),要如何变为奇异局势呢?假设 a < b
    < c,我们只要将 c 变为 a(+)b,即可,因为有如下的运算结果: a(+)b(+)(a(+)
    b)=(a(+)a)(+)(b(+)b)=0(+)0=0。要将c 变为a(+)b,只要从 c中减去 c-(
    a(+)b)即可。

        例1。(14,21,39),14(+)21=27,39-27=12,所以从39中拿走12个物体即可达
    到奇异局势(14,21,27)。

        例2。(55,81,121),55(+)81=102,121-102=19,所以从121中拿走19个物品
    就形成了奇异局势(55,81,102)。

        例3。(29,45,58),29(+)45=48,58-48=10,从58中拿走10个,变为(29,4
    5,48)。

        例4。我们来实际进行一盘比赛看看:
            甲:(7,8,9)->(1,8,9)奇异局势
            乙:(1,8,9)->(1,8,4)
            甲:(1,8,4)->(1,5,4)奇异局势
            乙:(1,5,4)->(1,4,4)
            甲:(1,4,4)->(0,4,4)奇异局势
            乙:(0,4,4)->(0,4,2)
            甲:(0.4,2)->(0,2,2)奇异局势
            乙:(0,2,2)->(0,2,1)
            甲:(0,2,1)->(0,1,1)奇异局势
            乙:(0,1,1)->(0,1,0)
            甲:(0,1,0)->(0,0,0)奇异局势
            甲胜。

    总结: 奇异局势(异或为0) 先手败   非奇异局势 先手胜

     

       HDU 1849 走棋游戏

       

    1、棋盘包含1*n个方格,方格从左到右分别编号为0,1,2,…,n-1;
    2、m个棋子放在棋盘的方格上,方格可以为空,也可以放多于一个的棋子;
    3、双方轮流走棋;
    4、每一步可以选择任意一个棋子向左移动到任意的位置(可以多个棋子位于同一个方格),当然,任何棋子不能超出棋盘边界;
    5、如果所有的棋子都位于最左边(即编号为0的位置),则游戏结束,并且规定最后走棋的一方为胜者。
    如果每次都是Rabbit先走棋


    Sample Input
    2
    3 5
    3
    3 5 6
    0

    Sample Output
    Rabbit Win!
    Grass Win!

     1 # include <iostream>
     2 # include <cstdio>
     3 # include <cmath>
     4 # include <algorithm>
     5 using namespace std ;
     6 
     7 int main()
     8 {
     9     int ans,n,a;
    10     while(scanf("%d",&n),n)
    11     {
    12         ans=0;
    13         while(n--)
    14         {
    15             scanf("%d",&a);
    16             ans ^= a;
    17         }    
    18         if(ans==0)  printf("Grass Win!
    ");
    19         else  printf("Rabbit Win!
    ");
    20     }    
    21     return 0;
    22 }    
    View Code

    HDU 1850

    桌子上有M堆扑克牌;每堆牌的数量分别为Ni(i=1…M);两人轮流进行;每走一步可以任意选择一堆并取走其中的任意张牌;桌子上的扑克全部取光,则游戏结束;最后一次取牌的人为胜者。
    现在我们不想研究到底先手为胜还是为负,我只想问大家:
    ——“先手的人如果想赢,第一步有几种选择呢?”

    求有几种方案将 非奇异局势 转化为 奇异局势

    Sample Input
    3
    5 7 9
    0

    Sample Output
    1

     1 # include <iostream>
     2 # include <cstdio>
     3 using namespace std ;
     4 
     5 int a[110] ;
     6 
     7 int main ()
     8 {
     9     int n ;
    10     while (scanf("%d" , &n) ) {
    11         if (n == 0)
    12            break ;
    13            
    14     int ans = 0 ;
    15     int sum = 0 ;
    16     
    17     int i ;
    18     for (i = 1 ; i <= n ; i++)
    19       {
    20         scanf("%d" , &a[i]) ;
    21         ans ^= a[i] ;
    22       }
    23     for (i = 1 ; i <= n ; i++)
    24     {
    25         if (a[i] > (ans ^ a[i]))
    26            sum++ ;
    27     }
    28     
    29     printf("%d
    " , sum) ;
    30     
    31     }
    32 }
    View Code

    14年省赛  路边骗局

    作为一个江湖骗子,night_watcher又在路边行骗了。现在他正在路边向路人介绍他的新游戏:
    有N堆石子 两个人轮流对其操作 。操作分为两步 第一步是每个人必须执行的:从某堆石子中取一部分(至少一个) 丢弃;第二步可以选择执行或不执行:从之前操作的那堆中拿一部分出来构成新堆。两个人轮流操作,不能操作的人被认为输。
    现在给出N堆石子每一堆的个数,假设每次都是路人先操作,且两人都足够聪明,请问路人能否取胜。

    样例输入
    3
    1 2 7
    样例输出
    Yes

     1 # include <cstdio>
     2 # include <algorithm>
     3 using namespace std ;
     4  
     5 int n ;
     6  
     7  
     8 int main ()
     9 {
    10     while (~scanf("%d" , &n))
    11     {
    12         int i , x ;
    13         int sum = 0 ;
    14         for (i = 1 ; i <= n ; i++)
    15         {
    16             scanf("%d", &x) ;
    17             sum ^= x ;
    18         }
    19         if (sum)
    20            printf("Yes
    ") ;
    21         else
    22            printf("No
    ") ;
    23          
    24     }
    25  
    26      
    27     return 0 ;
    28 }
    View Code

     

  • 相关阅读:
    窗体句柄总是0
    该函数设置由不同线程产生的窗口的显示状态。
    zip压缩解压缩 项目icsharpcode-SharpZipLib-e012155
    VS(Microsoft Visual Studio2010)工具打开项目所需的应用程序,出现未安装(.csproj)的应用程序的解决办法
    Cocoa深入学习:NSOperationQueue、NSRunLoop和线程安全
    iPhone 6出现后,如何将一份设计稿支持多个尺寸?
    iPhone/Mac Objective-C内存管理教程和原理剖析
    What does the “__block” keyword mean?
    UIView frame, bounds and center
    ARC __bridge modifiers demystified
  • 原文地址:https://www.cnblogs.com/mengchunchen/p/4490643.html
Copyright © 2020-2023  润新知