Nim游戏变种——取纽扣游戏

  (2016腾讯实习生校招笔试题)Calvin和David正在玩取纽扣游戏,桌上一共有16个纽扣,两人轮流来取纽扣,每人每次可以选择取1个或3个或6个(不允许不取),谁取完最后的纽扣谁赢。Cavin和David都非常想赢得这个游戏,如果Cavin可以先取,Cavin的必胜策略下第一步应该取
  A、1个
  B、3个
  C、6个
  D、Cavin没有必胜策略

  解析:这道题是Nim游戏的变种,Nim游戏是博弈论中最经典的模型(之一)。

  根据博弈论的性质:对于巴什博弈,存在必胜点必败点,是指在当前这个点上的先手玩家是“必胜”(指的是采取必胜策略下的必胜)还是必败。对于一个点,如果它的下一步全是必胜点,那么显然无论它如何走都是让对手进入必胜点,所以当前这个点就是必败点;如果下一步中存在一个必败点,那么当前这一步的玩家就可以选择让对手进入这个必败点的走法,所以当前这个点就是是必胜点

  对这题来说,显然0的时候是必败点;看1,只能选择拿走1个,变成0,0是必败点,1可以到达必败点,所以1是必胜点;然后看2,2的下一步只能选择拿走1个变成1,1是必胜点,2只能到达必胜点,所以2是必败点;3的下一步2和0,都是必败点,3可以到达必败点,所以3是必胜点.......同理,当推到16的时候,下一步有三种走法,分别是15,13,10,由前面推理过程,15和10都是必胜点,只有13是必败点。因此先手应该选择拿走3个棋子,让对手进入必败点。

  详细推理过程如下表(自上而下):

必败点 必胜点
0  
  1
2  
  3
4  
  5
  6
  7
  8
9  
  10
11  
  12
13  
  14
  15
  16

  我们还可以用程序验证我们的答案,源码如下:

#include "iostream"
#include "vector"
#include "string.h"
using namespace std;

int main()
{
    int button_num = 16;             // 共有16个纽扣
    int legal_get[3] = { 1,3,6 };    // 每人每次可以选择取1个或3个或6个
    bool win_point[17];              // 标识剩下i个纽扣的局面下,是否为必胜点
    int win_to_get[17];              // 必胜点下,采用必胜策略该取几个纽扣

    win_point[0] = false;                    // 0是必败点
    for (int i = 1; i <= button_num; i++)
    {
        win_point[i] = false;                // 初始化均为必败点
        for (int j = 0; j < 3; j++)
        {
            if (legal_get[j] <= i && !win_point[i - legal_get[j]])// 若下一步中存在一个必败点,则当前这个点就是是必胜点。
            {
                win_point[i] = true;
            }
            if (win_point[i])
            {
                win_to_get[i] = legal_get[j];    // 记录下必胜点下,采用必胜策略该取几个纽扣让对手进入必败点
                break;
            }
        }
    }
    if (win_point[button_num])    // 16为必胜点
    {
        cout << "Calvin有必胜策略,制胜的第一步应该取" << win_to_get[button_num] << "个纽扣" << endl;
    }
    else
        cout << "Cavin没有必胜策略" << endl;
    return 0;
}

  

  参考资料:http://www.guokr.com/blog/777525/

       https://www.zhihu.com/question/42098847/answer/93624200

时间: 2024-08-06 18:55:03

Nim游戏变种——取纽扣游戏的相关文章

[FAFU 1292]博弈论,组合游戏,取石游戏

http://acm.fafu.edu.cn/problem.php?id=1292 这题的原型就是<由感性认识到理性认识——透析一类搏弈游戏的解答过程>这篇论文介绍的取石游戏C 游戏C: 甲乙两人面对若干排石子,其中每一排石子的数目可以任意确定.例如图2所示的初始局面:共n=3排,其中第一排的石子数a1=7,第二排石子数a2=3,第三排石子数a3=3.两人轮流按下列规则取走一些石子,游戏的规则如下: Ø  每一步必须从某一排中取走两枚石子: Ø  这两枚石子必须是紧紧挨着的: Ø  如果谁无

292. Nim Game (取物游戏) by Python

292. Nim Game 题目: You are playing the following Nim Game with your friend: There is a heap of stones on the table, each time one of you take turns to remove 1 to 3 stones. The one who removes the last stone will be the winner. You will take the first

BZOJ 1874: [BeiJing2009 WinterCamp]取石子游戏 [Nim游戏 SG函数]

小H和小Z正在玩一个取石子游戏. 取石子游戏的规则是这样的,每个人每次可以从一堆石子中取出若干个石子,每次取石子的个数有限制,谁不能取石子时就会输掉游戏. 小H先进行操作,他想问你他是否有必胜策略,如果有,第一步如何取石子. N≤10 Ai≤1000 裸SG函数啊 然而我连SG函数都不会求了,WA了一会儿之后照别人代码改发现vis公用了... #include <iostream> #include <cstdio> #include <cstring> #includ

bzoj 1874 取石子游戏 题解 &amp;amp; SG函数初探

[原题] 1874: [BeiJing2009 WinterCamp]取石子游戏 Time Limit: 5 Sec  Memory Limit: 162 MB Submit: 334  Solved: 122 [Submit][Status] Description 小H和小Z正在玩一个取石子游戏. 取石子游戏的规则是这种,每一个人每次能够从一堆石子中取出若干个石子,每次取石子的个数有限制,谁不能取石子时就会输掉游戏. 小H先进行操作,他想问你他是否有必胜策略,假设有,第一步怎样取石子. In

bzoj 1874 取石子游戏 题解 &amp; SG函数初探

[原题] 1874: [BeiJing2009 WinterCamp]取石子游戏 Time Limit: 5 Sec  Memory Limit: 162 MB Submit: 334  Solved: 122 [Submit][Status] Description 小H和小Z正在玩一个取石子游戏. 取石子游戏的规则是这样的,每个人每次可以从一堆石子中取出若干个石子,每次取石子的个数有限制,谁不能取石子时就会输掉游戏. 小H先进行操作,他想问你他是否有必胜策略,如果有,第一步如何取石子. In

【BZOJ 1874】 [BeiJing2009 WinterCamp]取石子游戏

1874: [BeiJing2009 WinterCamp]取石子游戏 Time Limit: 5 Sec  Memory Limit: 162 MB Submit: 473  Solved: 186 [Submit][Status] Description 小H和小Z正在玩一个取石子游戏. 取石子游戏的规则是这样的,每个人每次可以从一堆石子中取出若干个石子,每次取石子的个数有限制,谁不能取石子时就会输掉游戏. 小H先进行操作,他想问你他是否有必胜策略,如果有,第一步如何取石子. Input 输

BZOJ 1874: [BeiJing2009 WinterCamp]取石子游戏(SG函数)

Time Limit: 5 Sec  Memory Limit: 162 MBSubmit: 871  Solved: 365[Submit][Status][Discuss] Description 小H和小Z正在玩一个取石子游戏. 取石子游戏的规则是这样的,每个人每次可以从一堆石子中取出若干个石子, 每次取石子的个数有限制,谁不能取石子时就会输掉游戏. 小H先进行操作,他想问你他是否有必胜策略,如果有 ,第一步如何取石子. Input 输入文件的第一行为石子的堆数N 接下来N行,每行一个数A

ZOJ 3964 Yet Another Game of Stones Nim游戏变种

ZOJ3964 解题思路 此题的题意比较容易理解,可以简单的看着 Nim 博弈的变种.但问题在于 Alice 对第 i 堆石子的取法必须根据 bi 确定.所以如果这个问题能够归结到正常的 Nim 博弈(取石子问题),则很容易解决. 考虑特判存在 bi=1 或 bi=2 的情况: 如果存在第 i 堆石子,其 ai 为奇数且 bi=2 ,则 Bob 必胜(Alice 在最优策略下无法取完该堆,但 Bob 可以). 如果存在 2 个及以上 bi=2 或 bi=1 且 ai>1 的情况,则 Bob 必胜

BZOJ 1874 取石子游戏 (NIM游戏)

题解:简单的NIM游戏,直接计算SG函数,至于找先手策略则按字典序异或掉,去除石子后再异或判断,若可行则直接输出. #include const int N=1005; int SG[N],b[N],hash[N],a[N],sum,tmp,i,j,n,m; void FSG(int s){ SG[0]=0; for(int i=1;i<=s;i++){ for(int j=1;b[j]<=i&&j<=m;j++)hash[SG[i-b[j]]]=i; for(int j