-
Notifications
You must be signed in to change notification settings - Fork 111
/
Copy pathsolution.cpp
94 lines (79 loc) · 3.93 KB
/
solution.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
/*
Assuming you know very well about finding grandy value of every state .
Now think about , a normal number 105 .
Its all divisors are
` 1, 3, 5, 7, 15, 21, 35, 105`
So , Our tower with height` (105)` can be broken into `105/Z=35(Y)` towers with all have equal size `Z=3` . So grandy value of a state is the mex of all the possible states which can be reached by a valid move (mex is the minimum state that is not possible to be reached from that particular state ) .
Here , when we will break a tower ` (105)` into `35(Y)` towers with size of all `3(Z)` .Then the grandy value of ` (105)` will be `G(105)`.
Where `G(105)` will be xor of all other grandy value of child towers of this father tower .
`G(105)` = `G(3)` ^ `G(3)`^`G(3)`^`G(3)`......... till `35(Y)` times . But we can easily catch that if `Y` is odd then no need to xor of all `Z` . It will be obviously only `G(3)` as rest of all are even in number ( canceled out by the xor ) .
So , `G(105)`= `G(3)` when , we will break it into `35(Y)` towers with size of all `3(Z)` only and not take other possible state's grandy vale .
But we can break it into its all divisors according to question . More fromally ,we have to take into account all possible state's grandy value .So use sieve to calculate all divisors of a number till `100000` previously. Now according to question grandy of `(105)` will be mex of the set { ` G(1), G(3),G( 5), G(7), G(15), G(21), G(35),G( 105 )` } and another fact is when `Y` will be even for any `Z` ,then Xor will be zero i.e. `G(Z) =0` . I hope all are clear . My pseudocode to find grandy recursively
*/
/// Nice problem based on grandy value
#include <bits/stdc++.h>
using namespace std;
const int mx=100001;
int grandy[mx];
vector<int>divisors[mx];
/// sieve of finding divisors of every number till mx
void sieve()
{
for(int i=2;i<mx;i++)
{
for(int j=i;j<mx;j=j+i)
{
divisors[j].push_back(i);/// all numbers (j) has a divisor i obviously
}
}
}
int sg(int X)
{
if(X==1)/// when the X is 1 there is no move ,grandy value is zero
{
return 0;
}
if(grandy[X]!=-1)/// if grandy value is already computed just return it , no need to calculate it
{
return grandy[X];
}
set<int>s;
for(int i=0;i<divisors[X].size();i++)
{
int Y=divisors[X][i];/// the tower will be broken down into Y towers ,Y>1
/// 1 is not included into any number's list of divisors during sieve ,so this Y will not be 1 never
int Z=X/Y;/// Each of Y towers has height of Z , may be also height of 1 which will terminate our program by finding our grandy value of every X,as X=1 is base condition .
if(Y%2==0) /// The number of towers are even then ,obviously its all tower's size will be canceled out as being of equal for Xor technique,grandy value will be 0 .
s.insert(0);
else /// The number of towers are odd,obviously take only grandy value of one tower's size (Z).rest of the towers are canceled out by each other for xor technique.
s.insert(sg(Z));
}
int mex=0;
while(s.find(mex)!=s.end()) mex++;///checking which smallest number not included in the set.
grandy[X]=mex;///assigning mex[not included minimum element in the set of grandy value of possible towers]
return grandy[X];/// return grandy
}
int main()
{
ios_base::sync_with_stdio(false);
cin.tie(NULL);
cout.tie(NULL);
sieve();
int n;
memset(grandy,-1,sizeof(grandy));/// initialize with -1
int tc;
cin>>tc;
while(tc--)
{
cin>>n;
int ans=0,h;
for(int i = 1; i <= n; ++ i)
{
cin>>h;
ans^=sg(h);
}
if(ans) cout<<"1"<<endl;/// First player won
else cout<<"2"<<endl;/// Second player won
}
return 0;
}