Revenge of Nim II

Problem Description
Nim is a mathematical game of strategy in which two players take turns removing objects from distinct heaps. On each turn, a player must remove at least one object, and may remove any number of objects provided they all come from the same heap.
---Wikipedia
Today, Nim takes revenge on you, again. As you know, the rule of Nim game is rather unfair, only the nim-sum (⊕) of the sizes of the heaps is zero will the first player lose. To ensure the fairness of the game, the second player has a chance to move some (can be zero) heaps before the game starts, but he has to move one heap entirely, i.e. not partially. Of course, he can’t move all heaps out, at least one heap should be left for playing. Will the second player have the chance to win this time?
Input
The first line contains a single integer T, indicating the number of test cases.
Each test case begins with an integer N, indicating the number of heaps. Then N integer Ai follows, indicating the number of each heap.
[Technical Specification]
1. 1 <= T <= 100
2. 1 <= N <= 1 000
3. 1 <= Ai <= 1 000 000 000 000
Output
For each test case, output “Yes” if the second player can win by moving some (can be zero) heaps out, otherwise “No”.
Sample Input
3
1
2
3
2 2 2
5
1 2 3 4 5
Sample Output
No
Yes
Yes

题目大意:

    Nim的复仇。有N堆石子,后手可以在游戏开始前拿走任意堆石子再进行Nim博弈,问后手是否有机会赢。

解题目标:

    先来复习一下Nim博弈。

    给定N堆石子,假设第i堆石子有a[i]个石头子,两个人轮流从某一堆中取任意多的物品,规定每一次至少取一个,多者不限,最后取光者胜。

    Nim博弈的常规解法:k=a[1]^a[2]^a[3]^....^a[n]

      若k==0 先手必输

      若k!=0  后手必输

    显然这个题的目标就是 找出石子堆集合的一个子集(子集元素个数大于等于2),使其元素异或之后为0即可。

解题思路:

    错误思路:

        做比赛的时候,想着用随机化算法来水过去,循环了350W次,提交了3遍。。 发现N<=1000,随机化算法不可能过。。

    正确思路:

        大牛的blog就是不一般。思路就一两句话,研究了整整一天,才勉强看懂。(一大牛题解就三个字:位运算。。Orz)

        根据官方题解的思路来看,要把这些数字看成一个二进制矩阵

        该矩阵具有一下性质:

        (1)每一个空都是1或0

        (2)相对于只有1或0两个值的矩阵元素,那么异或运算相当于其加减法:1+1=0  1+0=1  0+1=1  0+0=0  1-1=0  1-0=1  0-1=1  0-0=0

        再根据矩阵的秩与极大无关组的相关定理可知:若矩阵的秩rank<N,那么极大无关组的成员数要小于N,那么对于任意一个a[i](特别对于不属于极大无关组的a[i])都可以用极大无关组中的成员来进行表示。

        在根据性质:对于任意数a,a^a=0可知,当且仅当rank<N时,存在一个子集使其异或值为0。

解题方法:

    据说是高斯消元。将矩阵化成梯形,那么存在空行,即rank<N。

    ps:前面说了,异或在二进制中相当于加减。与矩阵的初等变化不矛盾。

Code:

 /*************************************************************************
> File Name: BestCode#16_1003.cpp
> Author: Enumz
> Mail: 369372123@qq.com
> Created Time: 2014年11月01日 星期六 17时44分16秒
************************************************************************/ #include<iostream>
#include<cstdio>
#include<cstdlib>
#include<string>
#include<cstring>
#include<list>
#include<queue>
#include<stack>
#include<map>
#include<set>
#include<algorithm>
#include<cmath>
#include<bitset>
#include<time.h>
#include<climits>
#define MAXN 3000
using namespace std;
long long a[MAXN];
int main()
{
int T;
cin>>T;
while (T--)
{
int N;
cin>>N;
for (int i=;i<=N;i++)
scanf("%I64d",&a[i]);
int row=,col=;
for ( ;row<=N&&col<=;col++,row++) //有数据范围可知,化成二进制之后,列数不会超过40
{
int tmp_row;
for (tmp_row=row;tmp_row<=N;tmp_row++) //找到当前列下的第一个不为零的值
if (a[tmp_row]&(1LL<<col)) //判断是否为1
break;
if (tmp_row==N+) //不存在为1的点,说明存在一个空行,rank--
{
row--;
continue;
}
swap(a[tmp_row],a[row]); //交换行
for (int i=tmp_row+;i<=N;i++)
if (a[i]&(1LL<<col))
a[i]^=a[row];
}
if (row<N)
printf("Yes\n");
else
printf("No\n");
}
return ;
}

    

HDU5088——Revenge of Nim II(高斯消元&矩阵的秩)(BestCoder Round #16)的更多相关文章

  1. 【bzoj3105】[cqoi2013]新Nim游戏 高斯消元求线性基

    题目描述 传统的Nim游戏是这样的:有一些火柴堆,每堆都有若干根火柴(不同堆的火柴数量可以不同).两个游戏者轮流操作,每次可以选一个火柴堆拿走若干根火柴.可以只拿一根,也可以拿走整堆火柴,但不能同时从 ...

  2. BZOJ 3105: [cqoi2013]新Nim游戏 [高斯消元XOR 线性基]

    以后我也要用传送门! 题意:一些数,选择一个权值最大的异或和不为0的集合 终于有点明白线性基是什么了...等会再整理 求一个权值最大的线性无关子集 线性无关子集满足拟阵的性质,贪心选择权值最大的,用高 ...

  3. BZOJ 3569: DZY Loves Chinese II [高斯消元XOR 神题]

    http://www.lydsy.com/JudgeOnline/problem.php?id=3569 题意:多次询问一个无向连通图当图中某k条边消失时这个图是否联通 强制在线 太神啦啦啦啦啦啦啦啦 ...

  4. ACM学习历程—HDU 3915 Game(Nim博弈 && xor高斯消元)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3915 题目大意是给了n个堆,然后去掉一些堆,使得先手变成必败局势. 首先这是个Nim博弈,必败局势是所 ...

  5. POJ 3185 The Water Bowls (高斯消元)

    题目链接 题意:翻译过来就是20个0或1的开关,每次可以改变相邻三个的状态,问最小改变多少次使得所有开关都置为0,题目保证此题有解. 题解:因为一定有解,所以我们可以正序逆序遍历两次求出较小值即可.当 ...

  6. A - The Water Bowls POJ - 3185 (bfs||高斯消元)

    题目链接:https://vjudge.net/contest/276374#problem/A 题目大意:给你20个杯子,每一次操作,假设当前是对第i个位置进行操作,那么第i个位置,第i+1个位置, ...

  7. Codeforces 832E Vasya and Shifts - 高斯消元

    题目传送门 快速的传送门I 快速的传送门II 题目大意 (题意比较复杂,请自行阅读原题) 可以将原题的字母都看成它们的在字符表中的下标,这样问题就变成给定$n$个$m$维向量$\vec{a_{1}}, ...

  8. hdu 3915 高斯消元

    http://acm.hdu.edu.cn/showproblem.php?pid=3915 这道题目是和博弈论挂钩的高斯消元.本题涉及的博弈是nim博弈,结论是:当先手处于奇异局势时(几堆石子数相互 ...

  9. 洛谷P3389 高斯消元 / 高斯消元+线性基学习笔记

    高斯消元 其实开始只是想搞下线性基,,,后来发现线性基和高斯消元的关系挺密切就一块儿在这儿写了好了QwQ 先港高斯消元趴? 这个算法并不难理解啊?就会矩阵运算就过去了鸭,,, 算了都专门为此写个题解还 ...

随机推荐

  1. CSS圆角效果

    看了院子里一篇关于CSS圆角技巧的文章,试了一下,觉得很好,贴出练习的代码.优秀文章链接: http://www.cnblogs.com/luluping/archive/2010/06/26/176 ...

  2. 【面试虐菜】—— JAVA面试题(3)

    1 throws与throw的区别 解析:throws和throw是异常处理时两个常见的关键字,初级程序员常常容易正确理解throw和throws的作用和区别,说明已经能比较深入理解异常处理.Thro ...

  3. Linux下C程序插入执行shell脚本

    1.system(执行shell命令) 相关函数 fork,execve,waitpid,popen表头文件 #include<stdlib.h>定义函数 int system(const ...

  4. c 指针兼容性问题

    指针兼容性问题: const指针不能赋值给非const指针. 非const指针可以赋值给const 指针,但前提是只是一层间接运算 Example: int *pt1; const *pt2; con ...

  5. 【转载】 使用Anemometer基于pt-query-digest将MySQL慢查询可视化

    原文地址:使用Anemometer基于pt-query-digest将MySQL慢查询可视化 作者:84223932 本文主要介绍使用Anemometer基于pt-query-digest将MySQL ...

  6. 为什么C/C++语言使用指针

    这是参加面试时,面试官问的一道开放性题目. 问题是:为什么C/C++语言使用指针? 这个问题一问出来,直接被面试官秒杀了,面试官大神,你怎么不按套路出牌啊? 说好的malloc和new的区别呢?说好的 ...

  7. STM32普通定时器实现延时函数

    /* SystemFrequency / 1000 1ms中断一次 * SystemFrequency / 100000 10us中断一次 * SystemFrequency / 1000000 1u ...

  8. php Linux安装

    参考地址:http://www.cnblogs.com/lianyue/p/3936728.html

  9. Win7超级终端查看单片机printf输出

    问题描述:     编写单片机C程序时,经常会用到printf输出信息进行查看,如何查看printf输出? 问题解决:     (1)编写单片机C程序     ucos是一个实时多任务操作系统,以上是 ...

  10. Java学习第二篇:类,对象,成员属性,成员方法,构造方法,类变量,类方法

    一.类的定义 一个全面的类定义是比较复杂的,  定义如下: