The 3rd Universal Cup 做题记录
Stage 0 - Stage 9:The 3rd Universal Cup 做题记录 (1)
Stage 10 - Stage 19:The 3rd Universal Cup 做题记录 (2)
Stage 0: Trial Contest ADFGHIJM
Stage 1: St. Petersburg ACDGHJKNO
Stage 2: Zielona Góra ACDEGH
Stage 4: Hongō BCDEKLN
Stage 5: Moscow ABEFGIKM
Stage 6: Osijek ABCDHIJKL
Stage 7: Warsaw ABEGKL
Stage 8: Cangqian BCDEFHIJLM
Stage 9: Xi'an AEFGHIJN
The 3rd Universal Cup. Stage 0: Trial Contest
A. Arrested Development
dp 可过。
D. Dihedral Group
一定是正序或倒序的连续区间。
F. Magic Bean
可以只交换一个环的 号和另一个环的 号。
G. Manhattan Walk
每个位置独立。设 为 去到终点的期望, 分别为 ,。如果 一定走 ,否则按等待时间划分走 还是 。
int n,m;
double dp[maxn][maxn],p;
void work(){
n=read();m=read();p=read();
dp[n][m]=0.0;
for(int i=n;i;i--){
for(int j=m;j;j--){
if(i==n&&j==m)continue;
else if(i==n)dp[i][j]=dp[i][j+1]+p/4.0;
else if(j==m)dp[i][j]=dp[i+1][j]+p/4.0;
else{
double a=dp[i][j+1],b=dp[i+1][j];
if(a>b)swap(a,b);
if(b>=a+p)dp[i][j]=a+p/4.0;
else{
dp[i][j]=a/2.0+(b-a)/(2*p)*(a+(b-a)/2.0)+(p-b+a)/(2*p)*b;
}
}
}
}
printf("%.10lf\n",dp[1][1]);
}
H. MountainCraft
查询区间并。线段树维护区间最小值和区间最小值个数。
I. Not Another Constructive!
在 A
处计算前缀 N
和后缀 C
的乘积。设 为前 个,前缀有 个,后缀有 个,能否有 个子序列。bitset 优化。
M. Training, Round 2
记录前 个题能否达到 。每个 只会更新一次。set 维护。
int n,a,b,ans;
set<int> s[maxn];
bool vis[maxn][maxn];
void work(){
n=read();a=read(),b=read();
s[0].insert(0);
for(int i=1;i<=n;i++){
int la=read()-a,ra=read()-a,lb=read()-b,rb=read()-b;
if(ra<0||rb<0)continue;
la=max(la,0ll),ra=min(ra,n),lb=max(lb,0ll),rb=min(rb,n);
vector<pii> tmp;
for(int j=la;j<=ra;j++){
auto it=s[j].lower_bound(lb);
while(it!=s[j].end()&&(*it)<=rb){
tmp.push_back({j,(*it)+1}),tmp.push_back({j+1,(*it)});
vis[j][(*it)]=1;
it=s[j].erase(it);
}
}
for(pii p:tmp)if(!vis[p.fi][p.se]){
ans=max(ans,p.fi+p.se);
s[p.fi].insert(p.se);
}
}
printf("%lld\n",ans);
}
The 3rd Universal Cup. Stage 1: St. Petersburg
A. Element-Wise Comparison
bitset 维护 。每 个点划一个段,统计跨过段的答案,维护一段的后缀 or。
int n,m,a[maxn],p[maxn],ans;
bitset<maxn> f[maxn],g[maxn],s;
void work(){
n=read();m=read();
for(int i=1;i<=n;i++)a[i]=read(),p[a[i]]=i;
for(int i=1;i<=n;i++)f[i].set(i);
for(int i=n;i;i--)f[p[i]]|=f[p[i+1]];
for(int i=n;i;i--)f[i]&=s,s.set(i);
for(int i=1;i<=n;i+=m){
for(int j=1;j<=m;j++)g[j].reset();
if(i+m-1>n)break;
g[m]=f[i+m-1];
for(int j=i+m-2;j>=i;j--)g[j-i+1]=g[j-i+2]&(f[j]<<(i+m-1-j));
s.set();
for(int j=i;j<=i+m-1&&j+m-1<=n;j++){
ans+=(s&g[j-i+1]).count();
if(j+m<=n)s&=f[j+m]>>(j-i+1);
}
}
printf("%lld\n",ans);
}
C. Cherry Picking
从大往小加,线段树维护区间前缀后缀和最大连续 。
D. Dwarfs' Bedtime
在 时刻先每个问一次,然后每隔 问一次,问到状态翻转就挂到 小时后连续的问。
G. Unusual Case
随机一个点开始找,当前找到 ,如果找不到新的出边,那么 连向 ,将路径改为 。据题解说是 的。
int n,m,k;
set<pii> s;
vector<int> e[maxn];
mt19937 rnd(time(0));
int id[maxn],tp;
bool vis[maxn];
void work(){
n=read();m=read();k=read();
for(int i=1;i<=m;i++){
int u=read(),v=read();
if(u>v)swap(u,v);
s.insert({u,v});
}
while(k--){
for(int i=1;i<=n;i++)vis[i]=0,e[i].clear();
id[tp=1]=rnd()%n+1;vis[id[1]]=1;
for(auto[x,y]:s)e[x].pb(y),e[y].pb(x);
while(tp<n){
if(rnd()&1)reverse(id+1,id+tp+1);
int u=id[tp];bool fl=0;
shuffle(e[u].begin(),e[u].end(),rnd);
for(int v:e[u])if(!vis[v]){
vis[v]=fl=1;id[++tp]=v;
break;
}
if(fl)continue;
for(int i=1;i<=tp;i++)if(id[i]==e[u].back()){
reverse(id+i+1,id+tp+1);
break;
}
}
for(int i=1;i<=tp;i++)printf("%lld ",id[i]);puts("");
for(int i=1;i<tp;i++)s.erase({min(id[i],id[i+1]),max(id[i],id[i+1])});
}
}
H. Page on vdome.com
答案为 。
J. First Billion
对 做 bitset 背包,还原方案时如果既可以选也可以不选,就随一个。取 。
int n,a[maxn];
int B=2500000;
bitset<2500000> dp[maxn];
mt19937 rnd(1);
void work(){
n=read();
for(int i=1;i<=n;i++)a[i]=read();
dp[0][0]=1;
for(int i=1;i<=n;i++){
dp[i]=(dp[i-1]<<(a[i]%B))|dp[i-1]|(dp[i-1]>>(B-a[i]%B));
}
while(1){
vector<int> id;
for(int i=n,s=0;i;i--){
if(dp[i-1][s]&&dp[i-1][(s-a[i]%B+B)%B]&&(rnd()&1))id.pb(i),s=(s-a[i]%B+B)%B;
else if(!dp[i-1][s])id.pb(i),s=(s-a[i]%B+B)%B;
}
int sum=0;for(int i:id)sum+=a[i];
if(sum==1e9){
printf("%lld ",id.size());for(int i:id)printf("%lld ",i);
return ;
}
}
}
也可以分为若干块,块内折半,块间选绝对值前 小合并。对不同块取不同 。
N. (Un)labeled graphs
个点维护每个点的编号,把 个点连成一条链。剩下 个点。取 为度数最大的点, 标记所有的 个点, 是唯一不连 的点,标记 和 个点的链头。
The 3rd Universal Cup. Stage 2: Zielona Góra
A. Interesting Paths
正反 dfs,找到可能在 上的点和边,答案为 。若一条路径经过 个之前没经过的点,则会经过 条之前没经过的边。一定能经过所有可能在 上的点和边。
C. Radars
能覆盖矩阵等价于能覆盖 个角落,状压 个角落是否被覆盖。
D. Xor Partitions
。按位拆开,。维护 ,表示当前 第 位为 的 之和。
E. Pattern Search II
。最大的 是 级别的。设 表示在 中从 出发向前向后能匹配多长的 ,需要多长的 ,仿 合并。
int n,m=30,ans=inf;
char s[maxn];
struct nd{
int len;
pii pre[maxn],suf[maxn];
}f[32];
nd merge(nd u,nd v){
nd res;res.len=u.len+v.len;
for(int i=1;i<=n;i++){
res.suf[i]=u.suf[i].fi==n+1?u.suf[i]:make_pair(v.suf[u.suf[i].fi].fi,u.len+v.suf[u.suf[i].fi].se);
res.pre[i]=v.pre[i].fi==0?v.pre[i]:make_pair(u.pre[v.pre[i].fi].fi,v.len+u.pre[v.pre[i].fi].se);
if(u.pre[i].fi==0&&v.suf[i+1].fi==n+1)ans=min(ans,u.pre[i].se+v.suf[i+1].se);
}
return res;
}
void work(){
scanf("%s",s+1);n=strlen(s+1);
if(n==1){puts("1");return ;}
for(int i=1;i<=n;i++){
if(s[i]=='a'){
f[1].pre[i]={i-1,1},f[1].suf[i]={i+1,1};
f[0].pre[i]=f[0].suf[i]={i,0};
}
else{
f[0].pre[i]={i-1,1},f[0].suf[i]={i+1,1};
f[1].pre[i]=f[1].suf[i]={i,0};
}
}
f[0].len=f[1].len=1;
for(int i=2;i<=m;i++)f[i]=merge(f[i-1],f[i-2]);
printf("%lld\n",ans);
}
G. Puzzle II
可以两步交换指定的两个数,换掉较少的颜色。序列的变化形如平移的形式,平衡树维护。
H. Weather Forecast
二分答案,取 前缀和的最长上升子序列和 比较。
剩下的待补。
The 3rd Universal Cup. Stage 4: Hongō
B. Black or White 2
令 ,。第 行隔一个填一个,之后插空一次填 和 。多出来的填 。
C. Contour Multiplication
为与 的所有 的共同系数。依次枚举 将 。
D. DRD String
设 为长为 且不存在 的方案数。前缀和转移。
E. Equally Dividing
只要 就有解。若 为偶,每列交替升序降序;否则 为奇,可以构造前 列填 。
K. Kth Sum
升序排序,二分答案 。对于每个 可以 计算 的答案。设阈值 ,前 个的贡献 计算。如果没超过 , 的 贡献 。后边只需要前 小的 对, 预处理。
int n,k,B;
int a[maxn],b[maxn],c[maxn];
priority_queue<pii> q;
int pos[maxn];
int d[maxn*200],tp;
int f(int x){
int res=0;
for(int i=1,j=n;b[i]+c[1]<=x&&i<=n;i++){
while(b[i]+c[j]>x)j--;
res+=j;
}
return res;
}
bool check(int x){
int sum=f(x-a[B]);
if(sum*B>=k)return 1;
for(int i=1;i<B;i++){
sum+=f(x-a[i]);
if(sum>=k)return 1;
}
for(int i=1,j=n;i<=tp&&j>B;i++){
while(j>B&&a[j]+d[i]>x)j--;
sum+=j-B;
if(sum>=k)return 1;
}
return 0;
}
void work(){
n=read();k=read();B=min(n,(int)sqrt(k/n)+1);
for(int i=1;i<=n;i++)a[i]=read();
for(int i=1;i<=n;i++)b[i]=read();
for(int i=1;i<=n;i++)c[i]=read();
sort(a+1,a+n+1),sort(b+1,b+n+1),sort(c+1,c+n+1);
for(int i=1;i<=n;i++)q.push({-(b[i]+c[pos[i]=1]),i});
while(!q.empty()&&tp<k/B){
d[++tp]=-q.top().fi;
int u=q.top().se;q.pop();
if(pos[u]<n)q.push({-(b[u]+c[++pos[u]]),u});
}
int l=0,r=inf,res=inf;
while(l<=r){
int mid=l+r>>1;
if(check(mid))res=mid,r=mid-1;
else l=mid+1;
}
printf("%lld\n",res);
}
L. Largest Triangle
取相邻的 海伦公式计算。
N. Number of Abbreviations
对于 要求 才能交换。
The 3rd Universal Cup. Stage 5: Moscow
A. Counting Permutations
模 和 相等的可以排列。
B. Bookshelf Tracking
在最后一次 R
操作前维护属于前半边还是后半边。
E. Building a Fence
分类讨论先做哪些,剩下的部分填到哪里。所有方式取 min。
F. Teleports
设 为区间 的答案,枚举 翻转,每次要么拆成更小的两个区间要么从相等长度的区间转移而来。先做第一种转移,第二种转移是同层最短路。
G. Exponent Calculator
。每个 要两次, 增大精度下降。计算 后平方 次。取 。
I. Marks Sum
在后 位一定存在前缀和 。 可以写为 ,,传 即可。
K. Train Depot
区间加,求区间 max。线段树合并。注意到修改的特殊性,合并时如果存在 就一定不优,返回 ,极大减小常数。否则需要递归到 且 再比较 和 ,理论复杂度依然正确但是常数极大。
M. Uniting Amoebas
答案为 。
The 3rd Universal Cup. Stage 6: Osijek
A. Coprime Array
特判 。除非 为奇 为偶,两步可行。否则多一个 。随机第一个数,失败时对于一个 的质因数 有 或 ,成功率为 。
B. Square Locator
设 B 坐标为 ,绕 A 顺时针或逆时针旋转 度得 D。联立解方程。
C. -is-this-bitset-
将 的 设为 ,能表示出所有 的倍数。对于 的 设 为 链上 的背包模 为 最小和为多少。记录修改的位置减少空间。
D. Cycle Game
警示:一个 的全黑矩形也不合法。
先最外层加一圈全为白。等价于时刻白色区域八连通且每个黑点都与一个白点八连通。线段树分治,初始一些边在一个前缀时间存在。到叶子时检查如果该点为黑与其八连通的黑点会不会无法与白点八连通。如果不删这个点,有些边重新合法。
H. Game Design
设 为前 层且有 条边跨过 的方案数。枚举 的出入边向前向后转移。
I. Geometry Hacking
在凹多边形是可能会死。构造 ,,,,面积都相等且最小。
J. Non-Interactive Nim
后手要使先手操作时存在二进制最高位的 只有一个。即后手操作时存在二进制最高位的 只有 个,将其中一个变为 。从高位往低位删即可。
K. String and Nails
每次删左下角即可,按 排序。
L. All-You-Can-Eat
维护当前选的 , 的集合, 的集合, 的集合。 一定合法,否则每次加入 时贪心调整三个集合。
The 3rd Universal Cup. Stage 7: Warsaw
A. Bus Analysis
代价 。dp 套 dp。内层 dp,设 为前 个点,代价为 最长向后多远。只有 的 dp 值有用。外层 dp 设 为前 个点, 的 dp 值分别在 。当 mn 增加时增加答案。除了最开始只有 的三元组有用。
int n,a[maxn],ans;
int dp[2][76][76][76],pw[maxn],f[6],cur;
void inc(int &u,int v){((u+=v)>=mod)&&(u-=mod);}
void work(){
n=read();
for(int i=1;i<=n;i++)a[i]=read();
pw[0]=1;for(int i=1;i<=n;i++)inc(pw[i]=pw[i-1],pw[i-1]);
dp[0][0][0][0]=1;
for(int i=1;i<=n;i++){
int d=a[i]-a[i-1];
mems(dp[i&1],0);
if(i==1){
int a=0,b=0,c=0;
int aa=max(0ll,a-d),bb=max(0ll,b-d),cc=max(0ll,c-d);
f[0]=aa,f[1]=bb,f[2]=cc,f[3]=f[4]=f[5]=0;
for(int j=0;j<5;j++){
f[j+1]=max(f[j+1],f[j]+20);
if(j+3<6)f[j+3]=max(f[j+3],f[j]+75);
}
for(int j=1;j<=5;j++)f[j]=max(f[j],f[j-1]);
inc(dp[i&1][f[0]][f[1]][f[2]],dp[cur][a][b][c]);
int p=0;while(!f[p])p++;
(ans+=dp[cur][a][b][c]*pw[n-i]%mod*p)%=mod;
inc(dp[i&1][f[p]][f[p+1]][f[p+2]],dp[cur][a][b][c]);
}
for(int a=0;a<=75;a++){
for(int b=a+20;b<=75;b++){
for(int c=b+20;c<=75;c++)if(dp[cur][a][b][c]){
int aa=max(0ll,a-d),bb=max(0ll,b-d),cc=max(0ll,c-d);
f[0]=aa,f[1]=bb,f[2]=cc,f[3]=f[4]=f[5]=0;
for(int j=0;j<5;j++){
f[j+1]=max(f[j+1],f[j]+20);
if(j+3<6)f[j+3]=max(f[j+3],f[j]+75);
}
for(int j=1;j<=5;j++)f[j]=max(f[j],f[j-1]);
inc(dp[i&1][f[0]][f[1]][f[2]],dp[cur][a][b][c]);
int p=0;while(!f[p])p++;
(ans+=dp[cur][a][b][c]*pw[n-i]%mod*p)%=mod;
inc(dp[i&1][f[p]][f[p+1]][f[p+2]],dp[cur][a][b][c]);
}
}
}
cur^=1;
}
printf("%lld\n",ans*2%mod);
}
B. Missing Boundaries
有不确定的端点先当成 。按左端点排序,记录最多最少能放多少 。
E. Express Eviction
将横纵坐标都小于等于 的障碍连边,等价于不存在左下边界到右上边界的路径。最小割。
G. Game of Geniuses
答案为最大的行的最小值。
K. Routing K-Codes
设 为 为根的和,, 为最大深度。换根 dp。
L. Random Numbers
期望为 。所以检查长度为 和 的区间。
The 3rd Universal Cup. Stage 8: Cangqian
B. Simulated Universe
设 为前 个, 之前有 个没匹配的祝福,向后能匹配 个祝福是否可行。把 压入状态,即 表示最大的 是多少。
C. Challenge NPC
构造二分图, 为左, 为右,。然后奇数为左向之前的右连边,偶数为右向除了 之前的左连边。
D. Puzzle: Easy as Scrabble
一个格子如果有两个方向的要求则为矛盾格子,最后为空。队列维护矛盾格子向后推。
E. Team Arrangement
注意到划分数不多,搜出来每组大小后贪心检查。按组的大小从小往大,在一个人的左端点处将他的右端点加入优先队列,对每个组优先取队列中最小的。
H. Permutation
二分,时刻维护当前区间次大值位置。如果次大值和最大值在同一边则用一次,否则两次。每次将 变为 用 次, 用 次,取 。
I. Piggy Sort
按位置之和排序,可以知道每张照片的先后顺序。 所以必然有一只猪出现在相邻照片的同一位置。枚举这个位置,检查是否存在这样一条直线经过 个点。每只猪只可能出现在一条直线上。
J. Even or Odd Spanning Tree
找到最小生成树,只替换一条边,维护奇偶边权的路径最大值。
L. Challenge Matrix Multiplication
每次找一条从 到 的路径, 减少 。对于一条路径, 的答案包含 的答案。从 到 依次 bfs 计算答案增量,每个点只经过一次。
int n,m;
vector<int> e[maxn],g[maxn];
int d[maxn];
int st[maxn],tp;
int ans[maxn];
bool vis[maxn];
queue<int> q;
void work(){
n=read();m=read();
for(int i=1;i<=m;i++){
int u=read(),v=read();
e[u].pb(v),g[u].pb(v);
d[u]++,d[v]--;
}
for(int t=1;t<=60;t++){
int p=0;for(int i=1;i<=n;i++)if(d[i]>0){p=i;break;}
if(!p)break;
tp=0;d[p]--;
while(d[p]>=0){
st[++tp]=p;
int v=e[p].back();e[p].pop_back();
p=v;
}
st[++tp]=p;d[p]++;
for(int i=1;i<=n;i++)vis[i]=0;
int num=0;
for(int i=tp;i;i--){
q.push(st[i]),vis[st[i]]=1;
while(!q.empty()){
int u=q.front();q.pop();;++num;
for(int v:g[u]){
if(!vis[v])q.push(v),vis[v]=1;
}
}
if(!ans[st[i]])ans[st[i]]=num;
}
}
for(int i=1;i<=n;i++)if(!ans[i])ans[i]=1;
for(int i=1;i<=n;i++)printf("%lld ",ans[i]);
}
The 3rd Universal Cup. Stage 9: Xi'an
A. An Easy Geometry Problem
差分,哈希 和反过来的 。线段树维护哈希值。二分答案。
E. Dominating Point
出度最大的点 为支配点。设 连向 , 连向 。如果存在 使得 ,那么 连向所有 和 ,,矛盾。如果 为空,无解。否则 组成的子图的支配点 也符合条件。如果 不是指向 所有点,递归下去得到 。否则指向 且属于 的部分的支配点符合条件。
F. An Easy Counting Problem
lucas 定理,等价于拆成 进制数下每一位的组合数之积。设 为填到前 位,组合数之积为 , 为一位时的答案。。满足结合律,快速幂优化。 为质数有原根,下标 改为 的 ,加法卷积。
H. Elimination Series Once More
从小到大加入,每次更新 个点。
I. Max GCD
质因数分解,枚举答案。可能贡献的 满足 是相邻的 的倍数,双指针最近的 。共 组。把 挂在 上,二位数点,做 插入 查询的分块。
J. Graph Changing
一次后无解,分讨 和 。