1191A - Tokitsukaze and Enhancement
Idea: tokitsukaze
Just enumerating the increment can pass.
However, by increasing at most $$$2$$$ points, the value of her HP can always become an odd number, and thus the highest possible level is either $$$A$$$ or $$$B$$$. We can just solve case by case.
#include<cstdio>
int x,i,A;char B;
char cal(int x){
x%=4;
if(x==0)return 'D';
if(x==1)return 'A';
if(x==2)return 'C';
return 'B';
}
int main(){
scanf("%d",&x);
A=0;
B=cal(x);
for(i=1;i<=2;i++){
if(cal(x+i)<B){
A=i;
B=cal(x+i);
}
}
printf("%d %c",A,B);
}
#include <bits/stdc++.h>
using namespace std;
int main()
{
int x;
cin>>x;
if(x%4==0) puts("1 A");
else if(x%4==1) puts("0 A");
else if(x%4==2) puts("1 B");
else if(x%4==3) puts("2 A");
return 0;
}
1191B - Tokitsukaze and Mahjong
Idea: tokitsukaze, 2014CAIS01
There are only two types of mentsus, so you can enumerate the mentsu you want her to form, and check the difference between that and those currently in her hand.
Alternatively, you can find out that the answer is at most $$$2$$$, since she can draw two extra identical tiles which are the same as one of those in her hand. You may enumerate at most $$$1$$$ extra tile for her and check if it can contribute to a mentsu. If she can't, the answer will be $$$2$$$.
#include <bits/stdc++.h>
using namespace std;
int main() {
int idx[257] = {}, c[3][9] = {}, ans = 2;
idx['m'] = 1;
idx['p'] = 2;
idx['s'] = 3;
for(int i = 0; i < 3; ++i) {
char buf[3];
scanf("%s", buf);
++c[idx[buf[1]] - 1][buf[0] - '1'];
}
for(int i = 0; i < 3; ++i)
for(int j = 0; j < 9; ++j) {
ans = min(ans, 3 - c[i][j]);
if(j + 2 < 9)
ans = min(ans, 3 - !!c[i][j] - !!c[i][j + 1] - !!c[i][j + 2]);
}
printf("%d\n", ans);
return 0;
}
//#pragma GCC optimize("O3")
#include <bits/stdc++.h>
using namespace std;
//defines
typedef long long ll;
typedef long double ld;
#define TIME clock() * 1.0 / CLOCKS_PER_SEC
#define prev _prev
#define y0 y00
//permanent constants
const ld pi = acos(-1.0);
const int day[12] = {31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};
const int digarr[10] = {6, 2, 5, 5, 4, 5, 6, 3, 7, 6};
const int dx[4] = {0, 1, 0, -1};
const int dy[4] = {1, 0, -1, 0};
const int dxo[8] = {-1, -1, -1, 0, 1, 1, 1, 0};
const int dyo[8] = {-1, 0, 1, 1, 1, 0, -1, -1};
const int alf = 26;
const int dig = 10;
const int two = 2;
const int th = 3;
const ll prost = 239;
const ll bt = 30;
const ld eps = 1e-7;
const ll INF = (ll)(1e18 + 239);
const int BIG = (int)(1e9 + 239);
const int MOD = 998244353;
const ll MOD2 = (ll)MOD * (ll)MOD;
//random
mt19937 rnd(239); //(chrono::high_resolution_clock::now().time_since_epoch().count());
//constants
const int M = (int)(2e5 + 239);
const int N = (int)(2e3 + 239);
const int L = 20;
const int T = (1 << 20);
const int B = (int)sqrt(M);
const int X = 1e4 + 239;
bool check(string a, string b, string c)
{
if (a == b && b == c) return true;
if (a > b) swap(a, b);
if (b > c) swap(b, c);
if (a > b) swap(a, b);
if (a[1] == b[1] && b[1] == c[1])
{
if (a[0] + 1 == b[0] && b[0] + 1 == c[0])
return true;
}
return false;
}
int32_t main()
{
#ifdef ONPC
freopen("input.txt", "r", stdin);
#endif
ios::sync_with_stdio(0); cin.tie(0); cout.tie(0);
string a, b, c;
cin >> a >> b >> c;
if (check(a, b, c))
{
cout << "0";
return 0;
}
vector<string> var;
for (int i = 1; i <= 9; i++)
{
string s = "";
s += (char)(i + '0');
var.push_back(s + "m");
var.push_back(s + "p");
var.push_back(s + "s");
}
for (string s : var)
{
if (check(s, a, b))
{
cout << "1";
return 0;
}
if (check(s, a, c))
{
cout << "1";
return 0;
}
if (check(s, b, c))
{
cout << "1";
return 0;
}
}
cout << "2";
return 0;
}
1190A - Tokitsukaze and Discard Items / 1191C - Tokitsukaze and Discard Items
Idea: tokitsukaze
The order of discarding is given, so we can simulate the process of discarding.
In each time, we can calculate the page that contains the first special item that has not been discarded, and then locate all the special items that need to be discarded at one time. Repeat this process until all special items are discarded.
Each time at least one item would be discarded, so the time complexity is $$$\mathcal{O}(m)$$$.
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int MAX=1e5+10;
ll p[MAX];
int main()
{
ll n,m,k;
scanf("%lld%lld%lld",&n,&m,&k);
for(int i=1;i<=m;i++) scanf("%lld",&p[i]);
int ans=0;
int sum=0;
int now=1;
while(now<=m)
{
ll r=((p[now]-sum-1)/k+1)*k+sum;
// cout<<"r:"<<r<<endl;
while(now<=m&&p[now]<=r)
{
sum++;
now++;
}
ans++;
}
printf("%d\n",ans);
return 0;
}
1190B - Tokitsukaze, CSL and Stone Game / 1191D - Tokitsukaze, CSL and Stone Game
Idea: tokitsukaze
Unless the first player must lose after the first move, the numbers of stones in these piles should form a permutation obtained from $$$0$$$ to $$$(n - 1)$$$ in the end, in order to ensure that there are no two piles include the same number of stones.
Let's use $$$cnt[x]$$$ to represent the number of piles which have exactly $$$x$$$ stones in the beginning. There are four cases that Tokitsukaze will lose after the first move:
- $$$cnt[0] > 1$$$;
- $$$cnt[x] > 2$$$ for some $$$x$$$;
- $$$cnt[x] > 1$$$ and $$$cnt[y] > 1$$$ for some $$$x$$$, $$$y$$$ ($$$x \neq y$$$);
- $$$cnt[x] > 1$$$ and $$$cnt[x - 1] > 0$$$ for some $$$x$$$.
If Tokitsukaze won't lose after the first move, then we only need to check the parity of the total number of stones that can be taken.
By the way, if you don't want to discuss the above four cases, you can just enumerate her first move.
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int MAX=1e5+10;
int a[MAX];
int main()
{
int n;
cin>>n;
map<int,int> mp;
int ok=1;
for(int i=1;i<=n;i++)
{
cin>>a[i];
mp[a[i]]++;
ok&=(mp[a[i]]<3); //situation 1: one >=3
}
int cnt=0;
for(auto it:mp)
{
if(it.second>=2) cnt++;
}
ok&=(cnt<=1);// situation 2: more than two >=2
for(auto it:mp)
{
if(it.second==2)
{
ok&=(mp.count(it.first-1)==0);// situation 3: reduce one but add another
if(it.first==0) ok=0;// situation 4: two pile is zero
}
}
if(!ok) return 0*puts("cslnb");
// normal situation
sort(a+1,a+1+n);
ll s=0;
for(int i=1;i<=n;i++)
{
assert(a[i]>=i-1);
s+=a[i]-(i-1);
}
s%=2;
return 0*puts(s?"sjfnb":"cslnb");
}
#include <bits/stdc++.h>
using namespace std;
const int maxn = (int)1e5 + 1;
int n, a[maxn], c[3];
void update(int x, int d) {
static map<int, int> ctr;
int &v = ctr[x];
--c[min(v, 2)];
v += d;
++c[min(v, 2)];
}
int main() {
scanf("%d", &n);
for(int i = 0; i < n; ++i) {
scanf("%d", a + i);
update(a[i], 1);
}
bool chk = 0;
for(int i = 0; !chk && i < n; ++i)
if(a[i]) {
update(a[i], -1);
update(a[i] - 1, 1);
chk |= !c[2];
update(a[i] - 1, -1);
update(a[i], 1);
}
if(chk) {
chk = 0;
for(int i = 0; i < n; ++i)
chk ^= (a[i] + i) & 1;
}
puts(chk ? "sjfnb" : "cslnb");
return 0;
}
1190C - Tokitsukaze and Duel / 1191E - Tokitsukaze and Duel
If a player can make a move from a situation to that situation again, this player will not lose. Except for some initial situations, one can move from almost every situation to itself. Based on these two conclusions, we can get a solution.
If Tokitsukaze cannot win after her first move, she cannot win in the future. In this case, we can quickly check if $$$k$$$ is so limited that she cannot win.
After her first move, it is possible that Quailty wins in the next move. If he cannot, in order to prevent failure, he can just leave the situation to his opponent by doing useless flipping and thus result in a draw.
Therefore, we can check if no matter how Tokitsukaze moves, Quailty has no chance to win after his first move. It can be easily checked linearly if we do some precalculation and enumerate Tokitsukaze's first move. Alternatively, we can find and check the pattern of initial situations in which Quailty can win.
#include<bits/stdc++.h>
using namespace std;
const int MAXN=100005;
int n,k;
char str[MAXN];
vector<int> loc[2];
int main()
{
scanf("%d%d%s",&n,&k,str);
for(int i=0;i<n;i++)
loc[str[i]-'0'].push_back(i);
if(loc[0].empty() || loc[1].empty())
return 0*printf("tokitsukaze\n");
int flag=0;
for(int l=0,r=k-1;r<n;l++,r++)
{
vector<int> tmp[2];
for(int i=0;i<2;i++)
{
if(loc[i].front()<l || loc[i].front()>r)
tmp[i].push_back(loc[i].front());
if(loc[i].back()<l || loc[i].back()>r)
tmp[i].push_back(loc[i].back());
auto itr=lower_bound(loc[i].begin(),loc[i].end(),l);
if(itr!=loc[i].begin())
tmp[i].push_back(*--itr);
itr=upper_bound(loc[i].begin(),loc[i].end(),r);
if(itr!=loc[i].end())
tmp[i].push_back(*itr);
sort(tmp[i].begin(),tmp[i].end());
}
if(tmp[0].empty() || tmp[1].empty())
return 0*printf("tokitsukaze\n");
flag|=(tmp[0].back()-tmp[0].front()>=k
|| tmp[1].back()-tmp[1].front()>=k);
}
return 0*printf("%s\n",(flag ? "once again" : "quailty"));
}
#include <bits/stdc++.h>
using namespace std;
const int maxn = (int)1e5 + 1;
int n, m, p[maxn], lft[maxn], rht[maxn];
char buf[maxn];
int main() {
scanf("%d%d%s", &n, &m, buf);
for(int i = 0, las[2] = {-1, -1}; i < n; ++i) {
int o = buf[i] - '0';
lft[i] = las[o ^ 1];
las[o] = i;
}
for(int i = n - 1, las[2] = {n, n}; i >= 0; --i) {
int o = buf[i] - '0';
rht[i] = las[o ^ 1];
las[o] = i;
}
bool fir = 0, sec = 1;
for(char op = '0'; op <= '1' && !fir; ++op) {
int sz = 0;
for(int i = 0; i < n; ++i)
if(buf[i] != op)
p[sz++] = i;
for(int L = 0, R = m - 1, pL = 0, pR = -1; R < n; ++L, ++R) {
for( ; pL < sz && p[pL] < L; ++pL);
for( ; pR + 1 < sz && p[pR + 1] <= R; ++pR);
if(pL == 0 && pR == sz - 1) {
fir = 1;
break;
}
int uL = p[pL ? 0 : pR + 1];
int uR = p[pR < sz - 1 ? sz - 1 : pL - 1];
int vL = uL > 0 ? 0 : min(rht[uL], uR + 1);
int vR = uR < n - 1 ? n - 1 : max(lft[uR], uL - 1);
sec &= uR - uL + 1 <= m || vR - vL + 1 <= m;
}
}
puts(fir ? "tokitsukaze" : (sec ? "quailty" : "once again"));
return 0;
}
#include<bits/stdc++.h>
using namespace std;
const int MAXN=1000005;
int n,k,T;
int a[MAXN],sum[MAXN];
int q_sum(int l,int r)
{
if(l>r)return 0;
return sum[r]-sum[l-1];
}
bool check_fir()
{
for(int i=1;i+k-1<=n;++i)
{
int lala=q_sum(1,i-1)+q_sum(i+k,n);
if(lala==0||lala+k==n)return true;
}
return false;
}
bool check_sec()
{
if(k*2<n||k==1)return false;
int len=n-k-1;
for(int i=2;i<=len;++i)
{
if(a[i]!=a[i-1]||a[n-i+1]!=a[n-i+2])return false;
}
if(a[len]==a[len+1]||a[n-len]==a[n-len+1]||a[1]==a[n])return false;
return true;
}
int main()
{
scanf("%d %d",&n,&k);
for(int i=1;i<=n;++i)
{
scanf("%1d",&a[i]);
sum[i]=sum[i-1]+a[i];
}
if(check_fir())
{
printf("tokitsukaze\n");
}
else
{
if(check_sec())
{
printf("quailty\n");
}
else
{
printf("once again\n");
}
}
return 0;
}
1190D - Tokitsukaze and Strange Rectangle / 1191F - Tokitsukaze and Strange Rectangle
Idea: tokitsukaze
For each strange rectangular area and its corresponding set of points, we only need to focus on the lowest $$$y$$$-coordinate $$$yB$$$, the leftmost $$$x$$$-coordinate $$$xL$$$ and the rightmost $$$x$$$-coordinate $$$xR$$$ of points in this set. Different sets must have different $$$yB$$$, $$$xL$$$, $$$xR$$$, so we can count them by enumerating these values.
Let's enumerate $$$yB$$$ first. Then, we need to list all the possible $$$x$$$-coordinates of points $$$(x_i, y_i)$$$ satisfying that $$$y_i \geq yB$$$ and mark every possible $$$x'$$$ satisfying there is a point $$$(x', yB)$$$. By doing so, we can make sure when enumerating $$$xL$$$ and $$$xR$$$, the requirement for $$$yB$$$ is met as well. However, enumerating forcibly, which is in time complexity $$$\mathcal{O}(n^3)$$$, is too slow to pass, so let's optimize the enumeration step by step.
Let's pick and sort the points $$$(x'_1, yB), (x'_2, yB), \ldots, (x'_m, yB)$$$ from left to right. Assuming that $$$x'_0 = 0$$$, and
Unable to parse markup [type=CF_MATHJAX]
is the leftmost point of them that is in the chosen set, we can count the number of aforementioned different $$$x$$$-coordinates in ranges $$$[x'_{j - 1} + 1, x'_j]$$$ and $$$[x'_j, \infty)$$$ and then count the number of possible pairs $$$(xL, xR)$$$ immediately. More specifically, let $$$cnt(l, r)$$$ be the number of different $$$x$$$-coordinates of points in the area $$$\lbrace (x, y) | l \leq x \leq r, y \geq yB \rbrace$$$, and we know the number of possible pairs $$$(xL, xR)$$$ is $$$cnt(x'_{j - 1} + 1, x'_j) \cdot cnt(x'_j, \infty)$$$. After precalculating $$$c(1, x)$$$ for each $$$yB$$$, we can reduce the time complexity into $$$\mathcal{O}(n^2)$$$.The very last step is using the trick of sweeping lines. We can enumerate $$$yB$$$ from highest to lowest, and during that process, use data structures to maintain possible $$$x$$$-coordinates. What we need to implement is to maintain a container, check if a coordinate is already in the container, add a coordinate to the container, and query the number of coordinates in a range. These requirements can be easily achieved by Fenwick tree, segment tree or others. With the last optimization, we can solve in time complexity $$$\mathcal{O}(n \log n)$$$.
By the way, there also exist solutions using other approaches, such as divide and conquer.
#include <bits/stdc++.h>
using namespace std;
#define mem(a,b) memset((a),(b),sizeof(a))
#define MP make_pair
#define pb push_back
#define fi first
#define se second
#define sz(x) (int)x.size()
#define all(x) x.begin(),x.end()
#define sqr(x) (x)*(x)
using namespace __gnu_cxx;
typedef long long ll;
typedef unsigned long long ull;
typedef pair<int,int> PII;
typedef pair<ll,ll> PLL;
typedef pair<int,ll> PIL;
typedef pair<ll,int> PLI;
typedef vector<int> VI;
typedef vector<ll> VL;
void go();
int main(){
#ifdef tokitsukaze
// freopen("TEST.txt","r",stdin);
#endif
go();return 0;
}
const int INF=0x3f3f3f3f;
const ll LLINF=0x3f3f3f3f3f3f3f3fLL;
const double PI=acos(-1.0);
const double eps=1e-6;
const int MAX=2e5+10;
const ll mod=998244353;
/********************************* head *********************************/
struct Fenwick_Tree
{
int bit[MAX];
int n;
void init(int _n){n=_n;mem(bit,0);}
int lowbit(int x){return x&(-x);}
void insert(int x,int v)
{
while(x<=n)
{
bit[x]+=v;
x+=lowbit(x);
}
}
int get(int x)
{
int res=0;
while(x)
{
res+=bit[x];
x-=lowbit(x);
}
return res;
}
int query(int l,int r)
{
return get(r)-get(l-1);
}
}tr;
VI x[MAX],y;
void go()
{
int n,i,tot,pre;
ll ans;
while(~scanf("%d",&n))
{
tr.init(n);
y.clear();
map<int,int> idx,idy,cntx;
tot=0;
for(i=0;i<n;i++)
{
int X,Y;
scanf("%d%d",&X,&Y);
if(!idy.count(Y))
{
x[tot].clear();
idy[Y]=tot++;
y.pb(Y);
}
x[idy[Y]].pb(X);
cntx[X]++;
}
tot=0;
for(auto it:cntx)
{
idx[it.fi]=++tot;
tr.insert(tot,1);
}
sort(all(y));
ans=0;
for(auto Y:y)
{
sort(all(x[idy[Y]]));
pre=1;
for(auto X:x[idy[Y]])
{
ans+=1LL*tr.query(pre,idx[X])*tr.query(idx[X],tot);
pre=idx[X]+1;
cntx[X]--;
if(cntx[X]==0)
{
cntx.erase(X);
tr.insert(idx[X],-1);
}
}
}
printf("%lld\n",ans);
}
}
#include <bits/stdc++.h>
using namespace std;
//priority_queue<int> q;
//priority_queue<int,vector<int>, greater<int> > q;
#include <ext/pb_ds/tree_policy.hpp>
#include <ext/pb_ds/assoc_container.hpp>
using namespace __gnu_pbds;
typedef tree<int, null_type, less<int>, rb_tree_tag, tree_order_statistics_node_update> less_rbtree;
typedef tree<int, null_type, greater<int>, rb_tree_tag, tree_order_statistics_node_update> greater_rbtree;
typedef tree<pair<int,int>, null_type, less<pair<int,int> >, rb_tree_tag, tree_order_statistics_node_update> multi_less_rbtree;
typedef tree<pair<int,int>, null_type, greater<pair<int,int> >, rb_tree_tag, tree_order_statistics_node_update> multi_greater_rbtree;
const int MAXN=300005;
less_rbtree div_tree;
long long ans;
struct node
{
int x,y;
}p[MAXN],temp[MAXN];
int n;
bool cmp1(const node & A,const node & B)
{
return A.x<B.x;
}
bool cmp2(const node & A,const node & B)
{
if(A.y!=B.y)return A.y>B.y;
return A.x<B.x;
}
void div_algorithm(int l,int r)
{
if(l>r)return;
div_tree.clear();
for(int i=l;i<=r;++i)
{
temp[i]=p[i];
}
int mid_line=p[(l+r)>>1].x;
long long preans=ans;
int lpos=-1,rpos=-1;
sort(temp+l,temp+r+1,cmp2);
for(int i=l;i<=r;++i)
{
div_tree.insert(temp[i].x);
if(temp[i].x<=mid_line)
{
if(lpos==-1||lpos<temp[i].x)
{
lpos=temp[i].x;
}
}
if(temp[i].x>=mid_line)
{
if(rpos==-1||rpos>temp[i].x)
{
rpos=temp[i].x;
}
}
if(i==r||temp[i+1].y!=temp[i].y)
{
long long lsum=div_tree.order_of_key(lpos+1);
long long Lsum=div_tree.order_of_key(mid_line+1);
long long rsum=div_tree.size()-div_tree.order_of_key(rpos);
long long Rsum=div_tree.size()-div_tree.order_of_key(mid_line);
if(lpos==-1)
{
ans+=Lsum*rsum;
}
else if(rpos==-1)
{
ans+=lsum*Rsum;
}
else
{
ans+=lsum*Rsum+Lsum*rsum-lsum*rsum;
}
lpos=-1;
rpos=-1;
}
}
int pin1=l,pin2=r;
while(p[pin1].x!=mid_line)++pin1;
while(p[pin2].x!=mid_line)--pin2;
div_algorithm(l,pin1-1);
div_algorithm(pin2+1,r);
}
int main()
{
scanf("%d",&n);
for(int i=1;i<=n;++i)
{
scanf("%d %d",&p[i].x,&p[i].y);
}
sort(p+1,p+1+n,cmp1);
div_algorithm(1,n);
printf("%I64d\n",ans);
return 0;
}
1190E - Tokitsukaze and Explosion
Idea: chenjb, Subconscious
It is obvious that we can binary search the answer because we can pull any line closer to $$$O$$$ and the situation won't change. Applying a binary search, we focus on if it is possible to set $$$m$$$ barriers with the same distance to $$$O$$$ and meet the requirement. By doing so, we can draw a circle whose center is $$$O$$$ such that the closest point to $$$O$$$ on each barrier is on the circumference. Then it becomes a classic problem — for each barrier we can get the angle range of its closest point on the circle, and we have to choose at most $$$m$$$ points on the circle to ensure that there is at least one point within each range.
This kind of greedy trick is very simple on a sequence. As to ranges on a sequence, you only have to sort all ranges in increasing order of left endpoint and take every necessary right endpoint. As to ranges on a circle, you can enumerate a position to break the circle into a sequence. But if you enumerate as the same way on the sequence, the time complexity will be $$$\mathcal{O}(n^2)$$$ which is not enough.
Instead, we can double and extend these ranges into $$$2 n$$$ ranges and then regard them as on a sequence. We can do some precalculation, such as if we want to choose a point to cover the $$$i$$$-th sorted range and other ranges in its right as many as possible, what is the next range that we cannot cover. Let's denote that as $$$f[i][0]$$$, and then we can calculate the next range after $$$2^p$$$ repeated steps from the $$$i$$$-th range as $$$f[i][p]$$$, which can be obtained from $$$f[f[i][p - 1]][p - 1]$$$. After preparation, we can enumerate the beginning position and use binary lifting method, each time in complexity $$$\mathcal{O}(\log m)$$$, to know that whether we can use $$$m$$$ steps to cover all the ranges.
Therefore, the total time complexity can be $$$\mathcal{O}(n \log m \log D)$$$, where $$$D$$$ is the precision requirement.
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <algorithm>
#define pi acos(-1.0)
#define LOG 20
#define eps 1e-9
using namespace std;
const int N = 500010;
const double twopi = 2*acos(-1.0);
int n,m,tot,fa[N*2][LOG];
bool nd[N*2];
double xx[N],yy[N],di[N];
struct Segment
{
double l,r; bool typ;
Segment(){}
Segment(double l,double r,bool typ):l(l),r(r),typ(typ){}
}q[N*2];
bool cmp(const Segment &a,const Segment &b){ if(a.typ!=b.typ) return a.typ<b.typ; return a.l<b.l; }
bool check(double cur)
{
tot = 0;
for(int i=1;i<=n;i++)
{
double deg = atan2(yy[i],xx[i]), p = acos(cur/di[i]);
q[++tot] = Segment(deg-p,deg+p,0);
q[++tot] = Segment(deg-p+twopi,deg+p+twopi,1);
}
sort(q+1,q+1+tot,cmp);
double mr = 2147483600;
memset(nd,1,tot+1);
for(int i=tot;i;i--)
{
if(q[i].r>=mr) nd[i] = 0;
else mr = q[i].r;
}
tot = 0;
for(int i=1;i<=n;i++) if(nd[i]) q[++tot] = q[i];
for(int i=1;i<=tot;i++) q[i+tot] = Segment(q[i].l+twopi,q[i].r+twopi,1);
tot <<= 1;
int j = 1;
for(int i=1;i<=tot;i++)
{
while(j<=tot&&q[i].r>=q[j].l) j++;
fa[i][0] = j;
}
for(int j=1;(1<<j)<=tot;j++)
for(int i=1;i<=tot;i++)
fa[i][j] = fa[fa[i][j-1]][j-1];
for(int i=1;(i<<1)<=tot;i++)
{
int p = i, tmp = 0;
for(int k=m;k;k>>=1,tmp++)
if(k&1) p = fa[p][tmp];
if(p==0||p>=i+(tot>>1)) return true;
}
return false;
}
int main()
{
scanf("%d%d",&n,&m);
double l = 0, r = 1e100;
for (int i=1;i<=n;i++) {
scanf("%lf%lf",&xx[i],&yy[i]);
di[i] = sqrt(xx[i]*xx[i]+yy[i]*yy[i]);
r = min(r, di[i]);
}
if(m>=n) {
printf("%.10f\n",r);
return 0;
}
while (r - l > eps) {
double mid = (l + r) / 2;
if (check(mid)) l = mid; else r = mid;
}
printf("%.10f\n", l);
return 0;
}
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef long double db;
const int MAXN=100005;
const int DEP=18;
vector<pair<int,int> > filter(int n,vector<pair<int,int> > &seg)
{
vector<pair<int,int> > now;
for(auto &t:seg)
{
t.second=(t.second+n-1)%n;
int l=t.first,r=t.second;
if(l<=r)
{
now.push_back(make_pair(l,-r));
now.push_back(make_pair(l+n,-(r+n)));
}
else now.push_back(make_pair(l,-(r+n)));
}
sort(now.begin(),now.end());
vector<pair<int,int> > stk;
for(auto &t:now)
{
while(!stk.empty() && stk.back().second<=t.second)
stk.pop_back();
stk.push_back(t);
}
vector<pair<int,int> > res;
for(auto &t:stk)if(t.first<n)
res.push_back(make_pair(t.first,-t.second));
return res;
}
int go[MAXN<<1][DEP];
int cal(int n,vector<pair<int,int> > &seg)
{
seg=filter(n,seg);
int m=(int)seg.size();
if(m<2)return m;
for(int i=0;i<m;i++)
{
int l=seg[i].first,r=seg[i].second;
seg.push_back(make_pair(l+n,r+n));
}
for(int i=0;i<2*m;i++)
go[i][0]=lower_bound(seg.begin(),seg.end(),make_pair(seg[i].second+1,0))-seg.begin();
go[2*m][0]=2*m;
for(int j=1;j<DEP;j++)
for(int i=0;i<=2*m;i++)
go[i][j]=go[go[i][j-1]][j-1];
int res=m;
for(int i=0;i<m;i++)
{
int t=i,cnt=0;
for(int j=DEP-1;j>=0;j--)
if(go[t][j]<i+m)
t=go[t][j],cnt+=(1<<j);
if(res>cnt+1)res=cnt+1;
}
return res;
}
const db PI=acos(-1.0);
ll x[MAXN],y[MAXN];
db dis[MAXN],ang[MAXN],angl[MAXN],angr[MAXN],anga[MAXN<<1];
bool check(int n,int m,db r)
{
for(int i=0;i<n;i++)
{
db t=acos(min(1.0L,r/dis[i]));
angl[i]=ang[i]-t,angr[i]=ang[i]+t;
if(angl[i]<0)angl[i]+=2*PI;
if(angr[i]>=2*PI)angr[i]-=2*PI;
anga[i<<1]=angl[i],anga[i<<1|1]=angr[i];
}
sort(anga,anga+2*n);
int k=unique(anga,anga+2*n)-anga;
vector<pair<int,int> > seg;
for(int i=0;i<n;i++)
{
int l=lower_bound(anga,anga+k,angl[i])-anga;
int r=lower_bound(anga,anga+k,angr[i])-anga;
seg.push_back({l,r});
}
return cal(k,seg)<=m;
}
int main()
{
int n,m;
scanf("%d%d",&n,&m);
for(int i=0;i<n;i++)
scanf("%lld%lld",&x[i],&y[i]);
db md=1e7;
for(int i=0;i<n;i++)
{
dis[i]=sqrtl(x[i]*x[i]+y[i]*y[i]);
ang[i]=atan2(y[i],x[i]);
if(ang[i]<0)ang[i]+=2*PI;
md=min(md,dis[i]);
}
if(md<1e-7)return 0*printf("%.12Lf\n",0.0L);
db tl=0,tr=1;
while(tr<md && check(n,m,tr))
tl=tr,tr=min(md,2*tr);
for(int _=0;_<25;_++)
{
db tm=(tl+tr)/2;
if(check(n,m,tm))tl=tm;
else tr=tm;
}
return 0*printf("%.12Lf\n",(tl+tr)/2);
}
1190F - Tokitsukaze and Powers
Idea: skywalkert
Firstly, let me briefly review this problem for you. Given integers $$$n$$$, $$$m$$$ and $$$p$$$ ($$$n > 0$$$, $$$m > 1$$$, $$$p \neq 0$$$), we denote $$$S = \lbrace x | x \in \mathbb{Z}, 0 \leq x < m, \gcd(m, x) = 1 \rbrace$$$ and $$$T = \lbrace p^e \bmod m | e \in \mathbb{Z}, e \geq 0 \rbrace$$$, where $$$\gcd(m, x)$$$ means the greatest common divisor of $$$m$$$ and $$$x$$$, and you are asked to pick up $$$n$$$ distinct integers from $$$(S - T)$$$, the set of elements in $$$S$$$ but not in $$$T$$$, or report that it is unachievable. Besides, there is an additional restriction, $$$m = q^k$$$ for a prime number $$$q$$$ and a positive integer $$$k$$$.
It is not difficult to show $$$|S|$$$, the size of $$$S$$$, equals to $$$\varphi(m) = m (1 - \frac{1}{q})$$$. When $$$\gcd(m, p) > 1$$$, there is only one element $$$p^0 \bmod m = 1$$$ in $$$T$$$ which is coprime to $$$m$$$, so in that case, $$$|S - T| = \varphi(m) - 1$$$. To output a solution, you can just enumerate the smallest integers and skip the integer $$$1$$$ and any multiples of $$$q$$$. In this process, you won't need to enumerate more than $$$(2 n + 1)$$$ integers, because $$$\frac{n}{1 - \frac{1}{q}} + 1 \leq 2 n + 1$$$.
When $$$\gcd(m, p) = 1$$$, we can observe that $$$T \subseteq S$$$. Let $$$|T|$$$ be $$$\lambda$$$, and we may conclude from Euler's totient theorem that $$$\lambda \leq \varphi(m)$$$, and even $$$\lambda | \varphi(m)$$$. When $$$n > \varphi(m) - \lambda$$$, there is no solution, so we need to calculate $$$\lambda$$$ in order to determine the existence of solutions.
To calculate $$$\lambda$$$, you can just enumerate all the divisors of $$$\varphi(m)$$$ and check them using Fast Power Algorithm, because the number of divisors is not too large due to the restrictions. Alternatively, you can do it more advanced and check more efficiently like the following.
lambda = phi(m)
for each x in prime factors of phi(m) :
if pow(p, lambda / x, m) == 1 :
lambda = lambda / x
The above approach requires $$$\mathcal{O}(\log \varphi(m))$$$ calls of Fast Power Algorithm. It is used more regularly when searching a random primitive root in modulo some integer.
When implementing Fast Power Algorithm, you may notice an issue that the modular multiplication may not be easy to code, when the 128-bit integer operations are not directly provided on this platform. In a precise way, you can implement another like Fast Multiplying Algorithm, though it is a bit slow. But if you believe you are lucky enough and you prefer using C/C++, you can try the following code in a convenient way.
typedef long long LL;
typedef long double LD;
LL mod_mul(LL x, LL y, LL mod) { // 0 <= x, y < mod
LL ret = x * y - (LL)((LD)x * y / mod + 0.5) * mod;
return ret < 0 ? ret + mod : ret;
}
No matter what approach you used, the factorization of $$$(q - 1)$$$ is inevitable, so the problem requires you to factorize a relatively large number (less than $$$10^{18}$$$) a bit quickly. For example, you can use Pollard's Rho Algorithm with Miller-Rabin Primality Test. By Birthday Paradox, which is a heuristic claim, when $$$x$$$ is not a prime, Pollard's Rho Algorithm can split $$$x$$$ into $$$u \cdot v$$$ in $$$\mathcal{O}(\sqrt{\min(u, v)}) = \mathcal{O}(x^{1 / 4})$$$ iterations. By the way, you can obtain $$$q$$$ from $$$m$$$ by enumerating possible $$$k$$$ from large to small, instead of factorization.
Let's go back to the case that $$$\gcd(m, p) = 1$$$ and $$$n \leq \varphi(m) - \lambda$$$. When any solution exists, we can observe that $$$\varphi(m) - \lambda \geq \frac{\varphi(m)}{2}$$$, so there may exist some solutions based on random distribution. That is, if you are able to check whether an integer is in $$$T$$$ or not, you can roughly pick $$$2 n$$$ integers in $$$S$$$ and check them. One approach is like the mechanism built in the problem checker, but as space is limited, I would not expend it at here. One thing you should be aware of is that you can't pick them without checking, as the probability of failure may be so large, like $$$\frac{1}{2}$$$. Moreover, you can't use Baby-Step-Giant-Step algorithm since its time complexity $$$\mathcal{O}(\sqrt{n m})$$$ is too slow.
If there exists at least one primitive root in modulo $$$m$$$, then we can construct the output integers. Since a primitive root $$$g$$$ can represent $$$S$$$ as $$$\lbrace g^e \bmod m | e \in \mathbb{Z}, 0 \leq e < \varphi(m) \rbrace$$$, we can reduce the problem into something about modular equation that only concerns the exponents. As the number of primitive root in modulo $$$m$$$ is $$$\varphi(\varphi(m))$$$, if the root exists, we can get a random primitive root easily. Now let's assume we find a primitive root $$$g$$$ and $$$p \equiv g^u \pmod{m}$$$. An integer $$$(g^v \bmod m)$$$ is in $$$T$$$ if and only if the equation $$$e u \equiv v \pmod{\varphi(m)}$$$ has a solution of $$$e$$$. It is easy to show, when $$$\gcd(\varphi(m), u) | v$$$, a solution of $$$e$$$ always exists, so the construction is quite straightforward.
The only case we left is when no primitive root exists in modulo $$$m$$$, which occurs when $$$m = 2^k$$$, $$$k \geq 3$$$. In this case, we cannot represent all the elements in $$$S$$$ as non-negative integer powers of one specific number, but we can find a pseudo-primitive root $$$g'$$$ to use its powers to represent all the elements in the form of $$$(4 t + 1)$$$ in $$$S$$$ (as the group $$$(S, \times_m)$$$, also known as $$$(\mathbb{Z}/m\mathbb{Z})^{\times}$$$, is isomorphic to a direct product of cyclic groups, $$$\mathrm{C}_2 \times \mathrm{C}_{2^{k - 2}}$$$). Besides, $$$\lbrace m - ({g'}^{e} \bmod m) | e \in \mathbb{Z}, 0 \leq e < 2^{k - 2} \rbrace$$$ can represent the rest numbers (in the form of $$$(4 t + 3)$$$) in $$$S$$$. As the product of two integers in the form of $$$(4 t + 3)$$$ is in the form of $$$(4 t + 1)$$$, we can discuss the parity of $$$e$$$ in expression $$$(p^e \bmod m)$$$ and check if an integer is in $$$T$$$, so the construction could work after a little modification. By the way, there exist many different construction methods for this case.
In summary, to solve this problem, we need to factorize $$$\varphi(m)$$$, calculate the order of $$$p$$$ in modulo $$$m$$$, find a primitive root $$$g$$$ (or a pseudo-primitive root $$$g'$$$) in modulo $$$m$$$ and enumerate some small integers (values or exponents) to construct a solution.
Time complexity: $$$\mathcal{O}((m^{1 / 4} + n) M(m))$$$, where $$$M(m)$$$ means the cost of one multiplication modulo $$$m$$$. Some slow solutions are accepted as well.
#include <bits/stdc++.h>
using namespace std;
typedef long long LL;
inline LL mod_add(LL x, LL y, LL mod) {
return (x += y) < mod ? x : x - mod;
}
inline LL mod_mul(LL x, LL y, LL mod) {
const int BLEN = __builtin_clzll(mod) - 1;
const LL BMSK = (1LL << BLEN) - 1;
LL ret = 0;
if(x < y)
swap(x, y);
while(y > 0) {
ret += x * (y & BMSK);
ret = ret < mod ? ret : ret % mod;
y >>= BLEN;
x <<= BLEN;
x = x < mod ? x : x % mod;
}
return ret;
/*
typedef long double LD;
LL ret = x * y - (LL)((LD)x * y / mod + 0.5) * mod;
return ret < 0 ? ret + mod : ret;
*/
}
inline LL mod_pow(LL x, LL k, LL mod) {
LL ret = mod > 1 ? 1 : 0;
for( ; k > 0; k >>= 1, x = mod_mul(x, x, mod))
if(k & 1)
ret = mod_mul(ret, x, mod);
return ret;
}
inline LL int_pow(LL x, int k, LL upp) {
LL ret = 1, lim = (++upp) / x;
for( ; k > 0; --k) {
if(ret > lim)
return upp;
ret *= x;
}
return ret;
}
inline bool miller_rabin(LL n) {
if(n == 2)
return 1;
if(n < 2 || !(n & 1))
return 0;
LL s, r;
for(s = 0, r = n - 1; !(r & 1); r >>= 1, ++s);
static const int ptot = 12, pr[ptot] = {2, 3, 5, 7, 11, 13, 17, 19, 23, 29, 31, 37};
for(int p : pr) {
if(p >= n)
break;
LL cur = 1, nxt = p;
for(LL k = r; k > 0; k >>= 1, nxt = mod_mul(nxt, nxt, n))
if(k & 1)
cur = mod_mul(cur, nxt, n);
for(int i = 0; i < s; ++i) {
nxt = mod_mul(cur, cur, n);
if(nxt == 1 && cur != 1 && cur != n - 1)
return 0;
cur = nxt;
}
if(cur != 1)
return 0;
}
return 1;
}
inline LL pollard_rho(LL n) {
static mt19937_64 rnd(996);
while(1) {
LL x = rnd() % (n - 3) + 2, y = x, c = rnd() % n, tim = 0, prd = 1;
for(LL i = 1, stp = 2; ; ++i) {
if(i == stp) {
y = x;
stp <<= 1;
}
if((x = mod_add(mod_mul(x, x, n), c, n)) == y)
break;
LL tmp = prd;
if((prd = mod_mul(prd, abs(y - x), n)) == 0)
return __gcd(tmp, n);
static const int maxt = 100;
if((++tim) < maxt)
continue;
if((prd = __gcd(prd, n)) > 1)
return prd;
tim = 0;
}
if(tim > 0 && (prd = __gcd(prd, n)) > 1)
return prd;
}
assert(0);
}
inline vector<pair<LL, LL> > factorize(LL n) {
vector<LL> vec;
queue<LL> que;
que.push(n);
while(!que.empty()) {
LL x = que.front();
que.pop();
for(LL y : vec)
for( ; x % y == 0; x /= y, vec.push_back(y));
if(x == 1)
continue;
if(miller_rabin(x)) {
vec.push_back(x);
} else {
LL y = pollard_rho(x);
que.push(y);
que.push(x / y);
}
}
sort(vec.begin(), vec.end());
vector<pair<LL, LL> > ret;
for(LL x : vec)
if(!ret.empty() && ret.back().first == x) {
++ret.back().second;
} else {
ret.push_back(make_pair(x, 1LL));
}
return ret;
}
const int maxn = (int)5e5 + 1, maxe = 60;
int n;
LL m, sp, pr, ex, phi, ord, gen;
void init() {
scanf("%d%lld%lld", &n, &m, &sp);
for(ex = maxe - 1; ex >= 1; --ex) {
if(m < (1LL << ex))
continue;
LL low = 2, upp;
for(upp = low << 1; int_pow(upp, ex, m) <= m; upp <<= 1);
while(low < upp) {
LL mid = (low + upp + 1) >> 1;
if(int_pow(mid, ex, m) <= m) {
low = mid;
} else {
upp = mid - 1;
}
}
if(int_pow(low, ex, m) == m) {
pr = low;
break;
}
}
assert(ex >= 1 && miller_rabin(pr));
phi = m - m / pr;
if(sp % pr == 0) {
ord = gen = -1;
return;
}
ord = phi;
vector<pair<LL, LL> > vec = factorize(pr - 1);
if(ex > 1)
vec.push_back(make_pair(pr, ex - 1));
for(pair<LL, LL> it : vec)
for( ; it.second > 0 && mod_pow(sp, ord / it.first, m) == 1; --it.second, ord /= it.first);
if(pr == 2 && m >= 8) {
assert((sp & 3) != 3 || !(ord & 1));
LL half_phi = phi / 2;
for(gen = 1; gen < m; gen += 4) {
bool chk = mod_pow(gen, half_phi, m) == 1;
for(pair<LL, LL> it : vec)
chk &= mod_pow(gen, half_phi / it.first, m) != 1;
if(chk)
break;
}
} else {
for(gen = 0; gen < m; ++gen) {
bool chk = mod_pow(gen, phi, m) == 1;
for(pair<LL, LL> it : vec)
chk &= mod_pow(gen, phi / it.first, m) != 1;
if(chk)
break;
}
}
}
void solve() {
if(n > phi - abs(ord)) {
puts("-1");
return;
}
if(ord == -1) {
LL val = 1;
for(int i = 1; i <= n; ++i) {
for(++val; val < m && val % pr == 0; ++val);
assert(val < m);
printf("%lld%c", val, " \n"[i == n]);
}
return;
}
if(pr == 2 && m >= 8 && (sp & 3) == 3) {
LL upp = phi / 2, stp = upp / (ord / 2);
for(LL e = 0, u = 1, v = sp; e < upp && n > 0; ++e, u = mod_mul(u, gen, m), v = mod_mul(v, gen, m))
if(e % stp > 0) {
printf("%lld%c", u, " \n"[!(--n)]);
if(n > 0)
printf("%lld%c", v, " \n"[!(--n)]);
}
} else {
LL upp = pr == 2 ? phi / 2 : phi, stp = upp / ord;
for(LL e = 0, v = 1; e < upp && n > 0; ++e, v = mod_mul(v, gen, m)) {
if(e % stp > 0)
printf("%lld%c", v, " \n"[!(--n)]);
if(n > 0 && pr == 2)
printf("%lld%c", m - v, " \n"[!(--n)]);
}
}
assert(n == 0);
}
int main() {
init();
solve();
return 0;
}
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef unsigned long long ull;
inline ll mul_mod(ll a, ll b, ll m)
{
a=(a%m+m)%m,b=(b%m+m)%m;
return ((a*b-(ll)(1.0L*a/m*b+0.5L)*m)%m+m)%m;
}
inline ll pow_mod(ll a,ll k,ll m)
{
ll res=1;
while(k)
{
if(k&1)res=mul_mod(res,a,m);
a=mul_mod(a,a,m);
k>>=1;
}
return res;
}
bool strong_pseudo_primetest(ll n,int b)
{
ll n2=n-1,res;
int s=0;
while(n2%2==0)n2>>=1,s++;
res=pow_mod(b,n2,n);
if((res==1)|| (res==n-1))return 1;
s--;
while(s>=0)
{
res=mul_mod(res,res,n);
if(res==n-1)return 1;
s--;
}
return 0;
}
bool is_prime(ll n)
{
static ll testNum[]={2,3,5,7,11,13,17,19,23,29,31,37};
static ll lim[]={4,0,1373653LL,25326001LL,25000000000LL,2152302898747LL,3474749660383LL,341550071728321LL,0,0,0,0};
if(n<2 || n==3215031751LL)return 0;
for(int i=0;i<12;i++)
{
if(n<lim[i])return 1;
if(!strong_pseudo_primetest(n,testNum[i]))return 0;
}
return 1;
}
inline ll func(ll x,ll n)
{
return (mul_mod(x,x,n)+1)%n;
}
ll pollard(ll n)
{
if(is_prime(n))return n;
if(~n&1)return 2;
for(int i=1;i<20;i++)
{
ll x=i,y=func(x,n),p=__gcd(y-x,n);
while(p==1)x=func(x,n),y=func(func(y,n),n),p=__gcd((y-x+n)%n,n)%n;
if(p==0 || p==n)continue;
return p;
}
return 1;
}
vector<ll> factor(ll n)
{
if(n==1)return {};
ll x=pollard(n);
if(x==n)return {n};
vector<ll> res=factor(x);
vector<ll> t=factor(n/x);
res.insert(res.end(),t.begin(),t.end());
return res;
}
vector<ll> solve_easy(int n,ll m)
{
vector<ll> res;
for(ll i=2;i<m && (int)res.size()<n;i++)
if(__gcd(i,m)==1)res.push_back(i);
return res;
}
bool dfs_sol(int i,int k,ull x,ull p,ull t)
{
if(i>k)return 1;
ull mask=(1ULL<<i)-1;
if((t&mask)==(x&mask) && dfs_sol(i+1,k,x,p*p,t))return 1;
if(((t*p)&mask)==(x&mask) && dfs_sol(i+1,k,x,p*p,t*p))return 1;
return 0;
}
vector<ll> solve_even(int n,ll m,ll p,int k)///m=2^k
{
vector<ll> res;
for(ll i=3;i<m && (int)res.size()<n;i+=2)
if(!dfs_sol(3,k,i,p,1))res.push_back(i);
return res;
}
void dfs_fac(int dep,ll val,vector<pair<ll,int> > &pcnt,vector<ll> &fac)
{
if(dep==(int)pcnt.size())
{
fac.push_back(val);
return;
}
for(int i=0;i<=pcnt[dep].second;i++)
{
dfs_fac(dep+1,val,pcnt,fac);
val*=pcnt[dep].first;
}
}
vector<ll> solve_root(int n,ll m,ll p,vector<ll> pm)
{
ll phi=m/pm[0]*(pm[0]-1);
vector<ll> pphi=factor(phi);
sort(pphi.begin(),pphi.end());
vector<pair<ll,int> > pcnt;
for(int i=0,j=0;i<(int)pphi.size();i=j)
{
while(j<(int)pphi.size() && pphi[i]==pphi[j])j++;
pcnt.push_back({pphi[i],j-i});
}
pphi.erase(unique(pphi.begin(),pphi.end()),pphi.end());
vector<ll> fac;
dfs_fac(0,1,pcnt,fac);
ll pmr=0;
while(++pmr)
{
bool isok=(__gcd(pmr,m)==1);
for(int i=0;i<(int)pphi.size() && isok;i++)
isok&=(pow_mod(pmr,phi/pphi[i],m)>1);
if(isok)break;
}
ll len=phi;
for(auto &t:fac)
if(pow_mod(p,t,m)==1)
len=min(len,t);
if(len==phi)return {};
vector<ll> res;
for(ll i=0;i<phi && (int)res.size()<n;i++)
if(i%(phi/len))res.push_back(pow_mod(pmr,i,m));
return res;
}
int main()
{
int n;
ll m,p;
scanf("%d%lld%lld",&n,&m,&p);
vector<ll> res;
if(__gcd(m,p)>1)
res=solve_easy(n,m);
else
{
vector<ll> pm=factor(m);
if(pm[0]==2 && (int)pm.size()>2)
res=solve_even(n,m,p,(int)pm.size());
else
res=solve_root(n,m,p,pm);
}
if((int)res.size()<n)
printf("-1\n");
else for(int i=0;i<n;i++)
printf("%lld%c",res[i]," \n"[i==n-1]);
return 0;
}
Can someone explain div2D
It is always optimal (except for on the first move) to take a stone from the smallest pile that isn't empty or one that would make a double. This ensures the sizes of two piles never "cross" and become the same in the middle of the game, which would imply that some player chose a losing move. In other words, the piles never change their order based on size. If there aren't any doubles to start with, this eventually ends up with the permutation described. Since there can't be any doubles after tokitsukaze's first move (or she would lose), do a bunch of casework for doubles on the first move. Then, count parity in total number of moves to reach the permutation.
A bunch of casework may be avoided. You can instead enumerate her first possible move and see if there are two piles of the same size.
Ohh that's brilliant!
Div1F:
It's actually pretty known that one can always set $$$g' = 5$$$ for $$$m = 2^k$$$; for $$$k \ge 2$$$, each odd remainder is uniquely represented as a number of the form $$$\pm 5^j$$$. (AFAIR this can be proved by induction.) Each number of the form $$$5^j$$$ is 1 mod 4, and each number of the form $$$-5^j$$$ is 3 mod 4.
Can anyone explain div2 C please
things you care about in any step is which item should i discard now and how many items discarded before this step and from which page should i start discarding this time, so we start with page part, it could be done using binary search, you want to find the minimum x which is satisfying this
where a is the next item to be discarded and b is number of discarded items before this step , by doing this every time you discard all items in current page, start shifting items and do it again until you finish all the items
we have to keep track of number of numbers discarded. let's denote it by cnt
Now we can start the process of discarding the elements. The position of the current element p[i] on the page can be calculated as pos = ( p[i]- cnt)%k; if(pos == 0) pos = k; Now delete all the elements with p[ j ] — p[ i ] <= k — pos for j>i increment cnt for every discarded number and increment ans for each iteration.
can someone explain me the approach on how to come up with the formula in div2 C ll r=((p[now]-sum-1)/k+1)*k+sum;
it just a sort of enumerating the right border to me, for p[now] , its right border is the current pages + deleted number ? forgive my poor eng
I am a bit confused with test 43 of div2D (Tokitsukaze, CSL and Stone Game):
2
1 1
So player 1 will take a stone, hence we have (0 1). Then player 2 will take another stone, so now we have (0 0). So player 2 loses (because two piles are the same number) and player 1 wins. However the answer gives player 2 as the winner. Am I missing something?
Yes. You missed the first player's name.
I don't understand. What do you mean?
When I checked that test case, I found the answer(winner) is sjfnb, who is the first player.
Ok I got it. I mistyped something in my code. Thanks.
I am a bit confused with div2D (Tokitsukaze, CSL and Stone Game): 4 1 1 1 1 it is the first peopeo win
In that case the second player wins.
why
Because the first player cannot remove any stone — whatever stone the player removes, the state of stone piles will be like this :
0 1 1 1(1 0 1 1, 1 1 0 1, 1 1 1 0 are all same)
and because there are three piles that contain same number of stone, the first player loses.
Because no matter how the first player removes the stones, there are always two piles contain the same number of stones ($$$1\ 1$$$)
a
i personally felt that VASYA is much better than TOKITSUKAZE.
In Div2C for test case :-
15 3 5
3 15 9
Shouldn't the answer be 3 and not 2 (editorial code gives 2 as answer)? We first discard 3,then 9 and then 15. All 3 of them are on different pages at all times.
Edit:- I am sorry, I didn't see the constraints that input should be in ascending order.
you may try 15 3 5 3 9 15
for the second line is in ascending-order
So what should be the answer ? It should be 3 right?
yeah
Your code/Editorial code gives answer as 2.
I mean the input you offer is illegal (since the second line has to be ascending order)
Please someone explain 1191B solution by skywalkert
$$$c[i][j]$$$ means the number of tiles with suit $$$i$$$ and digit $$$j$$$, which is $$$\leq 3$$$.
If you want Toki to form a triplet with suit $$$i$$$ and digit $$$j$$$, it will need $$$(3 - c[i][j])$$$ extra tiles.
If you want Toki to form a sequence $$$[(i, j), (i, j + 1), (i, j + 2)]$$$, it will require at least one tile in each type.
!!c[i][j]
in C/C++ can be explained as(c[i][j] == 0) == 0
orc[i][j] != 0
, which is a boolean value.How does the second player win with [4 4 4] as the given array in Div2 D
No matter how the first player moves, there will 2 piles of equal count of stones after his move
In Problem 1C:
at the beginning I got a wrong understanding on problem
and thinks that in each step we can choose a range and xor 1 on each bit
in this case I don't know how to work and just write an random solution
but I got wa17. (I don't know how did it passed 16 pretests xd
after read problem again I got real meaning
but I have notime to code and have to just fix my random solution and submit it
and in the end I got an FST on test 59 : (
I added a brute force part on my random solution and passed system test.
I have no idea how to hack it or could prove it works.
here is the link: 56941913, hack is welcomed :p
and, what if the problem is xor 1 but not (or 1 / and 0) ?
could this be solved by random / other solution?
Oh, in fact I forget to bruteforce on another side
this is the new link: 56980588
what I am think about is :
is there any test that have only(few) valid movement
and surely not only on sides but could be anywhere?
Sure, here you are. 2 moves lead to draw out of 100002 possible and both moves are pretty far from the start. I'm sure a test with lower probability of success exists but this is enough.
thank you, but could you please check my new comment?
in fact I constructed similar test in my code
(it's just I forget to spj another side : (
it's mainly I found out it's hard to build test that is draw and not on sides
so I question this :p
Ok, that's trickier :D
I am a bit confused with div2d, if the piles are (0 1), the second player wins but this doesn't look to fit in any of the four cases described
I finally understand. this falls in the case of the number of movements to obtain the permutation 0, .., n-1
For future contests, please do not make contestants write out long, random strings when outputting answers. Otherwise, I found (div. 2) contest enjoyable, with a nice problem balance.
Thank you for your advice:3
Hello) Can you please explain me task 1191D (Tokitsukaze, CSL and Stone Game). Exactly why we need to do s+=a[i]-(i-1) instead of just s+=a[i] ? I really can't understand this moment.
They are basically computing the sum of the stones on all piles, minus 0+1+2+..+n-1,the total number of stones remaining when the game is over.
Is it because after taking 0+1+2...+n-1 stones we will have at least two heaps(piles) with simmilar number of stones? I do not understand why we need to minus exactly this numbers.
The idea is, that when the game ends, the number of stones in each pile will be 0, 1, 2, ... n-1, since there cannot be two equally sized piles. Before the game ends each player is forced to take some stone (doesn't really matter which), without making two piles equal. So basically it is as if there was a single pile with SUM(piles)-(0+1+2+..+n-1) stones in it.
thanks a lot)
You're welcome!
Can someone please explain 1190D strange rectangle problem... The editorial seems quite confusing. Thanks in advance :D
Shouldn't I check all powers of prime factors of $$$\varphi(m)$$$ here?..
In case that $$$x^k | \varphi(m)$$$, where $$$x$$$ is a prime and $$$k$$$ is an integer greater than $$$1$$$, you may assume that
prime factors of phi(m)
in the mentioned (roughly written) pseudocode includes $$$x$$$ at least $$$k$$$ times, in other words,lambda = lambda / x
may execute $$$k$$$ times or more.Anyway, you are right as well.
for problem D (div2) does anyone know a good source for game theory or how to be better at it
Please someone explain DIV2 D. I can't understand what to do after checking the 4 cases given in editorial.
In div2E ..in input case 4 1 0 0 1 1 ans should be quality as it doesn't matter which card Tokitsukaze changes..quality will have always 1 card to change and therefore win..Please explain
You can choose '0' to '0', or '1' to '1'. So it is a draw.
In Tokitsukaze and Discard Items The following code gives correct answer on literally any other online IDE, but not on codeforces judge.
The test case it is failing is
(Gives 0 for the Codeforces judge while the correct answer is 2)
64 bit integer in c++ is long long not long. Maybe the problem is caused by it?
Can someone please explain 1191E — Tokitsukaze and Duel
How is the second inspection of the solution(by winterzz1) judged?
Sorry, my English is not that good.
Thanks in advance :D
In div2E .. in input case 4 1 0 0 1 1 ans should be quality as it doesn't matter which card Tokitsukaze changes..quality will have always 1 card to change and therefore win..Please explain
suppose in the first step tokistuke changes it to 1011 now quality should change it to 1111 for victory. why is he choosing for a draw??
Also can anybody tell the logic how to tackle?? i read the editorial but still was not able to figure it out.
You can choose '0' to '0', or '1' to '1'. So it is a draw.
Prime number and root are good things! Without them problems like 1190F will become so difficult and complex. :(
https://codeforces.net/contest/1190/submission/57910087 can someone help me out with my soln I am getting wrong ans on 31st test case