题意:

有1~n,n个数字,两个人轮流操作,每一次一个人可以拿一个数字$x$,之后$x, x^2, x^3....x^t$全都被删掉。

给定n,问最优策略下谁赢。

解法:

考虑SG函数,可以注意到题目中取走$x$后,$x^2,x^3...$不可以取,类似石子合并问题。

对于1~n的数字可以分为两类:

  1.不存在$x^t, t>1$,可以视为一堆只有一块石头的石子堆,$SG(1) = 0$。

  2.存在$x^t, t>1$。

对于第一种情况,直接记录有多少个x满足条件即可,分奇偶讨论。

对于第二种情况,显然有$t<=30$,可以注意到最终$SG$值和$x$无关,这样打表预处理$sg(t)$表示$x^1,x^2...x^t$

对应的$SG$值,打表直接用$O(2^30)$状压即可(注意到$sg(t)<=30$,所以用char类型的sg数组即可节省空间)。

实际上有效的状态并不多,所以只要几秒钟(意外的快)。

最后求NIM和即可。

总效率$O(\sqrt{n}*logn)$

 #include <iostream>
#include <cstdio>
#include <cstring> #define N 1000010
#define LL long long using namespace std; bool flag[N];
int n;
int SG[]={,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,}; int main()
{
scanf("%d",&n);
int sum=n;
int ans=;
for(LL i=;i*i<=n;i++)
{
if(flag[i]) continue;
int tmp=;
for(LL x=i;x<=n;x*=i)
{
if(x*x<=n) flag[x]=;
tmp++;
}
ans^=SG[tmp];
sum-=tmp;
}
ans ^= sum&;
if(ans==) puts("Petya");
else puts("Vasya");
return ;
}

最新文章

  1. C++STL -- vector实现
  2. unslider.js 实现移动web轮播
  3. 学习Word2vec
  4. mybatis中使用使用模块化sql
  5. Maven-通过命令操作maven项目
  6. HBase 压缩算法设置及修改
  7. LabVIEW串口通信
  8. 1、java基础回顾与加强
  9. ASP.NET 运行机制续(完结)
  10. 一个Hibernate小程序
  11. (原)opencv直线拟合fitLine
  12. UISearchDisplayController UISearchBar
  13. 第一百三十二节,JavaScript,封装库--下拉菜单
  14. IOS开发-UI学习-UIImageView控件
  15. Matrix Again(最大费用最大流)
  16. 理解性能的奥秘——应用程序中慢,SSMS中快(4)——收集解决参数嗅探问题的信息
  17. [shiro学习笔记]第四节 使用源代码生成Shiro的CHM格式的API文档
  18. jsonp原理,封装,应用(vue项目)
  19. java 类、方法、代码块修饰式关键字总结
  20. SQL Injection-Http请求的参数中对特殊字符的处理

热门文章

  1. 日志打印longging模块(控制台和文件同时输出)
  2. Android Volley分析(一)——结构
  3. mdadm
  4. Laravel建站02--配置Laravel
  5. Centos7安装配置ansible运维自动化工具
  6. h5的复制功能
  7. tween用户使用指南
  8. Aspose.cells 读取Excel表中的图片问题
  9. alsa 编程
  10. UVa 11572 唯一的雪花(优化策略)