ACdream 1112 Alice and Bob(素筛+博弈SG函数)
Time Limit:3000MS Memory Limit:128000KB 64bit IO Format:%lld & %llu
Description
Here is Alice and Bob again !
Alice and Bob are playing a game. There are several numbers.
First, Alice choose a number n.
Then he can replace n (n > 1) with one of its positive factor but not itself or he can replace n with a and b. Here a*b = n and a > 1 and b > 1.
For example, Alice can replace 6 with 2 or 3 or (2, 3).
But he can't replace 6 with 6 or (1, 6). But you can replace 6 with 1.
After Alice's turn, it’s Bob's turn. Alice and Bob take turns to do so. Who can’t do any replace lose the game.
Alice and Bob are both clever enough. Who is the winner?
Input
This problem contains multiple test cases. The first line contains one number n(1 ≤ n ≤ 100000).
The second line contains n numbers.
All the numbers are positive and less than of equal to 5000000.
Output
For each test case, if Alice can win, output “Alice”, otherwise output “Bob”.
Sample Input
2
2 2
3
2 2 4
Sample Output
Bob
Alice
题意:给定n堆石子,每次可以按照规则将该堆石子的个数分为两堆或者将该堆石子的个数减少,
谁不能继续操作了谁就输。
题解:规则如下,若n=a*b且a>1,b>1,可以将石子分为a,b两堆;或者可以减少为a或b。
比如n=6,可以分为(2,3),2,3这三种情况,但是不能分为(1,6),1,6这三种情况。
和HDU3032是一个问题,就是一个取石子问题,这里的8=2*2*2就是那道题里的3,这里的
6=2*3就是那道题里的2。如果对这道题理解感到困难可以先阅读我的上上篇博客:
http://www.cnblogs.com/Ritchie/p/5627242.html
也就是说这道题是对每个数的素因子个数进行操作,即素数可以看做是一个1。
设一个数x=a1^r1*a2^r2*...*an^rn;设sum = r1+r2+...+rn.
然后所有的情况就可以表示为:
(1,sum-1),(2,sum-2),...(sum/2,sum-sum/2)或者(1),(2),...(n-1)
然后在计算sum的时候我们可以这样计算。
设一个数为x,他的最小的素因子为y.则sum[x] = sum[x/y] + 1;
所以我们在素筛打表的时候要记录每个数的最小素因子以用于上述公式。
#include <iostream>
#include <cstring>
#include <cstdio>
using namespace std;
#define maxn 5000050
int prime[maxn],cnt=;
bool isprime[maxn];
int m[maxn];
int a[maxn];
int sg[];
void getprime()
{
memset(isprime,,sizeof(isprime));
memset(m,,sizeof(m));
memset(a,,sizeof(a));
for(int i=; i<=maxn; i++)
{
if(!isprime[i])
{
prime[cnt++]=i;
for(int j=i+i; j<=maxn; j+=i)
{
isprime[j]=;
if(!m[j]) m[j]=i;
}
m[i]=i;
}
}
for(int i=; i<=maxn; i++)
a[i]=a[i/m[i]]+;
}
void getsg()
{
memset(sg,,sizeof(sg));
sg[]=;
sg[]=;
bool vis[]; //放到循环里定义也一样需要每次都初始化
for(int i=; i<=; i++)
{
memset(vis,,sizeof(vis)); //注意每次都要初始化
for(int j=; j<i; j++) //取石子
vis[sg[j]]=;
for(int j=; j<i; j++) //拆分
vis[sg[j]^sg[i-j]]=;
for(int x=;; x++)
if(!vis[x])
{
sg[i]=x;
break;
}
}
}
void get()
{
getsg();
getprime();
}
void solve()
{
int n;
while(scanf("%d",&n)!=EOF)
{
int data,ans=;
for(int i=; i<n; i++)
{
scanf("%d",&data);
ans^=sg[a[data]];
}
if(ans)
printf("Alice\n");
else
printf("Bob\n");
}
}
int main()
{
get();
solve();
}
ACdream 1112 Alice and Bob(素筛+博弈SG函数)的更多相关文章
- ACdream 1112 Alice and Bob (sg函数的变形+素数筛)
题意:有N个数,Alice 和 Bob 轮流对这些数进行操作,若一个数 n=a*b且a>1,b>1,可以将该数变成 a 和 b 两个数: 或者可以减少为a或b,Alice先,问谁能赢 思路 ...
- Alice and Bob HDU - 4111 (SG函数)
Alice and Bob are very smart guys and they like to play all kinds of games in their spare time. The ...
- ACdream 1112 Alice and Bob (博弈&&素数筛选优化)
题目链接:传送门 游戏规则: 没次能够将一堆分成两堆 x = a*b (a!=1&&b!=1)x为原来堆的个数,a,b为新堆的个数. 也能够将原来的堆的个数变成原来堆的约数y.y!=x ...
- S-Nim HDU 1536 博弈 sg函数
S-Nim HDU 1536 博弈 sg函数 题意 首先输入K,表示一个集合的大小,之后输入集合,表示对于这对石子只能去这个集合中的元素的个数,之后输入 一个m表示接下来对于这个集合要进行m次询问,之 ...
- hdu 4111 Alice and Bob(中档博弈题)
copy VS study 1.每堆部是1的时候,是3的倍数时输否则赢: 2.只有一堆2其他全是1的时候,1的堆数是3的倍数时输否则赢: 3.其他情况下,计算出总和+堆数-1,若为偶数,且1的堆数是偶 ...
- hdu 3032(博弈sg函数)
题意:与原来基本的尼姆博弈不同的是,可以将一堆石子分成两堆石子也算一步操作,其它的都是一样的. 分析:由于石子的堆数和每一堆石子的数量都很大,所以肯定不能用搜索去求sg函数,现在我们只能通过找规律的办 ...
- Light OJ 1199 - Partitioning Game (博弈sg函数)
D - Partitioning Game Time Limit:4000MS Memory Limit:32768KB 64bit IO Format:%lld & %llu ...
- LightOJ 1315 - Game of Hyper Knights(博弈sg函数)
G - Game of Hyper Knights Time Limit:1000MS Memory Limit:32768KB 64bit IO Format:%lld & ...
- Light OJ 1296 - Again Stone Game (博弈sg函数递推)
F - Again Stone Game Time Limit:2000MS Memory Limit:32768KB 64bit IO Format:%lld & %llu ...
随机推荐
- 从svn服务器自动同步到另一台服务器
需求场景 A commit B post-commit C (workstation) --------------> (svn server) ---------------------> ...
- CSS文字排版
一.font-size 我来试一试:为第一段中的“胆小如鼠”设置字号为:20px,字体颜色为:red. <!DOCTYPE HTML> <html> <head> ...
- RedGate .NET Reflector注册问题(反注册)
Reflector分为桌面版和VS集成版本,当我们使用注册机注册的时候如果注册了Standvard版本,那么我们的VS就不能集成查看,也不能Debug,那么这 显然不是我们想要的,我们会选择重新注册, ...
- C++处理一个动态规划的问题
嗯哼,别人问的问题,看的我也头晕,百度了一下动态规划,看了看才想起来该怎么做,今天写了写代码,实现了~ 要求是递归,动态规划,想了想这种方法也是最简单的~ 所谓动态规划:把多阶段过程转化为一系列单阶段 ...
- linux下使用 Tomcat 的几个坑
总结:用sudo su - 后的身份启动tomcat,可选用 bin下的 ./catalina.sh run命令以显示启动过程中可能的报错信息 1.普通用户是无法使用0~1023的熟知端口的,需要 ...
- 标题右边10px位置紧跟发布时间
一个ul列表,拥有若干li,内容是新闻标题,标题右边10px位置紧跟发布时间,当标题过长需要控制标题width,需要兼容ie6,不能用max-width h4{font-size:14px;heigh ...
- iOS @try @catch简单应用举例
- linux ftp命令(转)
此命令需要安装ftp, yum install ftp 1. 连接ftp服务器 格式:ftp [hostname| ip-address]a)在linux命令行下输入: ftp 192.168.1.1 ...
- Linux中iptables设置详细(转)
无论如何,iptables是一个需要特别谨慎设置的东西,万一服务器不在你身边,而你贸然设置导致无法SSH,那就等着被老板骂吧,呵呵... 以下内容是为了防止这种情况发生而写的,当然很初级,不过一般服务 ...
- Silverlight实例教程 - 自定义扩展Validation类,验证框架的总结和建议(转载)
Silverlight 4 Validation验证实例系列 Silverlight实例教程 - Validation数据验证开篇 Silverlight实例教程 - Validation数据验证基础 ...