【GDOI2017 day1】取石子游戏 线段树+区间合并
题面
如果给你一棵有根树,树根为 1,并且树的每个结点上有一个权值,现在我想知道每个点,除它所在子树以外的结点权值集合的 mex,怎么做呢?
在这里,mex 是定义在集合上的函数,mex(S) 表示 S 这个集合中,最小的非负整数喔。
对于 20% 的数据:N ≤ 500, T ≤ 20
另外 50% 的数据:N ≤ 100000, T ≤ 5
最后 30% 的数据:N ≤ 1000000, T ≤ 1
100
\(O(nlogn)\)
考虑把树上问题变为序列上的问题:
每次询问相当于是抠掉dfs序列中的一段区间,然后询问前缀后缀;
考虑给序列复制一遍,那么前缀和后缀拼起来就变成了一个区间。
然后就变成了序列上的问题了。
【清华集训2014】mex
有一个长度为n的数组{a1,a2,...,an}。m次询问,每次询问一个区间内最小没有出现过的自然数。
我们构出权值主席树,每一位表示这一数字至此最后出现的位置。
那么显然有种二分的做法。
同时,离线可以把主席树变成线段树。
Back to Problem
然后这道题就等价于上述的那道题了。
\(O(n)\)
枚举i从0到n,考虑那些树上结点的答案是当前枚举的i。
我们给所有权值=i的结点做个LCA,这个LCA到根节点的路径上未赋值的答案都为i。
然后利用并查集和tarjan求LCA,这是可以做到\(O(n)\)(\(O(n*\alpha)\))
Code
正确性未知,可能会被卡常的\(O(nlogn)\)做法。
#include<bits/stdc++.h>
#define ll long long
#define fo(i,x,y) for(int i=x;i<=y;i++)
#define fd(i,x,y) for(int i=x;i>=y;i--)
using namespace std;
const int inf=0x7fffffff;
const char* fin="2.in";
const char* fout="2.out";
const int maxn=1000007,maxm=maxn*2,maxt=maxn*4;
int n,m,fi[maxn],la[maxm],ne[maxm],tot,num,a[maxn],w[maxm];
int b[maxn][2],hd,tl,dfn[maxn],ffn[maxn],si[maxn];
int c[maxt];
struct Q{int l,r,id;}q[maxn];
bool cmp(const Q &a,const Q &b){return a.r<b.r;}
void modify(int l,int r,int t,int v,int vv){
int mid=(l+r)/2;
if (l==r){c[t]=vv;return;}
if (v<=mid) modify(l,mid,t*2,v,vv);
else modify(mid+1,r,t*2+1,v,vv);
c[t]=(c[t*2]>c[t*2+1]?c[t*2+1]:c[t*2]);
}
int query(int l,int r,int t,int v){
int mid=(l+r)/2;
if (l==r) return l;
if (c[t*2]<v) return query(l,mid,t*2,v);
else return query(mid+1,r,t*2+1,v);
}
void add_line(int a,int b){
tot++;
ne[tot]=fi[a];
la[tot]=b;
fi[a]=tot;
}
int read(){
int x=0;
char ch=getchar();
while (ch<'0' || ch>'9') ch=getchar();
while (ch>='0' && ch<='9') x=x*10+ch-'0',ch=getchar();
return x;
}
void write(int x){
int w[20];
w[0]=0;
while (x) w[++w[0]]=x%10,x/=10;
fd(i,w[0],1) putchar(w[i]+'0');
}
void bfs(int v){
hd=tl=0;
b[++tl][0]=v;
b[tl][1]=0;
while (hd++<tl)
for(int k=fi[b[hd][0]];k;k=ne[k])
if (la[k]!=b[hd][1]) b[++tl][0]=la[k],b[tl][1]=b[hd][0];
}
void getdfn(){
bfs(1);
fd(i,tl,1){
int x=b[i][0],y=b[i][1];
si[x]=1;
for(int k=fi[x];k;k=ne[k]) if (la[k]!=y) si[x]+=si[la[k]];
}
fo(i,1,tl){
int x=b[i][0],y=b[i][1];
dfn[x]=ffn[x]=ffn[y]+1;
ffn[y]+=si[x];
}
}
void pre(){
scanf("%d%d",&n,&m);
tot=0;
memset(fi,0,sizeof fi);
memset(ffn,0,sizeof ffn);
fo(i,1,n) a[i]=read();
fo(i,1,m){
int j=read(),k=read();
add_line(j,k);
add_line(k,j);
}
}
int ans[maxn];
void solve(){
fo(i,1,n) w[dfn[i]]=w[dfn[i]+n]=a[i]+1;
fo(i,1,4*n) c[i]=0;
fo(i,1,n) q[i].l=dfn[i]+si[i],q[i].r=dfn[i]+n-1,q[i].id=i;
sort(q+1,q+n+1,cmp);
int j=1;
fo(i,1,n){
while (j<=q[i].r) modify(1,maxn,1,w[j],j),j++;
ans[q[i].id]=query(1,maxn,1,q[i].l)-1;
}
fo(i,1,n)write(ans[i]),putchar(' ');
printf("\n");
}
int main(){
freopen(fin,"r",stdin);
freopen(fout,"w",stdout);
int t=read();
while (t--){
pre();
getdfn();
solve();
}
return 0;
}
【GDOI2017 day1】取石子游戏 线段树+区间合并的更多相关文章
- HDU 3911 线段树区间合并、异或取反操作
题目:http://acm.hdu.edu.cn/showproblem.php?pid=3911 线段树区间合并的题目,解释一下代码中声明数组的作用: m1是区间内连续1的最长长度,m0是区间内连续 ...
- POJ 3667 Hotel(线段树 区间合并)
Hotel 转载自:http://www.cnblogs.com/scau20110726/archive/2013/05/07/3065418.html [题目链接]Hotel [题目类型]线段树 ...
- HDU 3911 Black And White(线段树区间合并+lazy操作)
开始以为是水题,结果...... 给你一些只有两种颜色的石头,0为白色,1为黑色. 然后两个操作: 1 l r 将[ l , r ]内的颜色取反 0 l r 计算[ l , r ]内最长连续黑色石头的 ...
- HYSBZ 1858 线段树 区间合并
//Accepted 14560 KB 1532 ms //线段树 区间合并 /* 0 a b 把[a, b]区间内的所有数全变成0 1 a b 把[a, b]区间内的所有数全变成1 2 a b 把[ ...
- 【bzoj2653】middle 可持久化线段树区间合并
题目描述 一个长度为n的序列a,设其排过序之后为b,其中位数定义为b[n/2],其中a,b从0开始标号,除法取下整.给你一个长度为n的序列s.回答Q个这样的询问:s的左端点在[a,b]之间,右端点在[ ...
- 【bzoj1858】[Scoi2010]序列操作 线段树区间合并
题目描述 lxhgww最近收到了一个01序列,序列里面包含了n个数,这些数要么是0,要么是1,现在对于这个序列有五种变换操作和询问操作: 0 a b 把[a, b]区间内的所有数全变成0 1 a b ...
- hdu 3308(线段树区间合并)
LCIS Time Limit: 6000/2000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Total Submis ...
- 【BZOJ3638】Cf172 k-Maximum Subsequence Sum 线段树区间合并(模拟费用流)
[BZOJ3638]Cf172 k-Maximum Subsequence Sum Description 给一列数,要求支持操作: 1.修改某个数的值 2.读入l,r,k,询问在[l,r]内选不相交 ...
- 【bzoj3638】Cf172 k-Maximum Subsequence Sum 模拟费用流+线段树区间合并
题目描述 给一列数,要求支持操作: 1.修改某个数的值 2.读入l,r,k,询问在[l,r]内选不相交的不超过k个子段,最大的和是多少. 输入 The first line contains inte ...
随机推荐
- <Python基础>类和对象(初级)---烧开水的例子
''' 类:模板(模子) 类的名称:类名(人) 类的属性:一组数据(年龄,身高) 类的方法:进行操作的方法(走,跑,吃,喝) 对象:实体 类的抽象:把现实中的物品用类去表示 ''' #创建一个类 cl ...
- vue表格之@row-click="handleSelect" 与setCurrentRow
作用:表格行点击触发的事件 注意与@change.@selection-change事件的区分 <el-table ref="RoomTable" @row-click=&q ...
- 使用Colaboratory的免费GPU训练神经网络
1 Colaboratory 介绍 Colaboratory 是一个 Google 研究项目,旨在帮助传播机器学习培训和研究成果.它是一个 Jupyter 笔记本环境,不需要进行任何设置就可以使用,并 ...
- hive设置列头(永久模式)
到hive目录下的hive-site <property> <name>hive.cli.print.header</name> <value>true ...
- There is no public key available for the following key IDs:3B4FE6ACC0B21F32
ubuntu 运行完sudo apt-get update之后,提示 W: There is no public key available for the following key IDs: 3B ...
- ssh 免密码登入
1.普通免密码登入 (1) 生成秘钥 [root@vick ~]# ssh-keygen -t rsa Generating public/private rsa key pair. Enter f ...
- java虚拟机(十一)--GC日志分析
GC相关:java虚拟机(六)--垃圾收集器和内存分配策略 java虚拟机(五)--垃圾回收机制GC 打印日志相关参数: -XX:+PrintGCDetails -XX:PrintGCTimestam ...
- leyou_03_cors解决ajax的跨域请求问题
1.为什么会有跨域问题 因为跨域问题是浏览器对于ajax请求的一种安全限制:一个页面发起的ajax请求,只能是与当前页域名相同的路径,这能有效的阻止跨站攻击. 因此:跨域问题 是针对ajax的一种限制 ...
- vbox虚拟机复制&&虚拟机指定静态IP
一.复制镜像(假设源镜像已经用桥接方式,可以访问互联网). 注意需要重新生成mac地址 二.复制完成,启动复制好的镜像(注意,此时的镜像无法联网) vi /etc/udev/rules.d/70-pe ...
- tornado nginx 同源(Access-Control-Allow-Origin)错误处理记录
如果以下两个步骤操作完还是没有解决问题,可能是因为程序出了别的错误,因为后来的调试中不论出什么问题它都是会返回同源错误..呵呵哒 1.在nginx.conf里配置server,在location里添加 ...