题面

如果给你一棵有根树,树根为 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】取石子游戏 线段树+区间合并的更多相关文章

  1. HDU 3911 线段树区间合并、异或取反操作

    题目:http://acm.hdu.edu.cn/showproblem.php?pid=3911 线段树区间合并的题目,解释一下代码中声明数组的作用: m1是区间内连续1的最长长度,m0是区间内连续 ...

  2. POJ 3667 Hotel(线段树 区间合并)

    Hotel 转载自:http://www.cnblogs.com/scau20110726/archive/2013/05/07/3065418.html [题目链接]Hotel [题目类型]线段树 ...

  3. HDU 3911 Black And White(线段树区间合并+lazy操作)

    开始以为是水题,结果...... 给你一些只有两种颜色的石头,0为白色,1为黑色. 然后两个操作: 1 l r 将[ l , r ]内的颜色取反 0 l r 计算[ l , r ]内最长连续黑色石头的 ...

  4. HYSBZ 1858 线段树 区间合并

    //Accepted 14560 KB 1532 ms //线段树 区间合并 /* 0 a b 把[a, b]区间内的所有数全变成0 1 a b 把[a, b]区间内的所有数全变成1 2 a b 把[ ...

  5. 【bzoj2653】middle 可持久化线段树区间合并

    题目描述 一个长度为n的序列a,设其排过序之后为b,其中位数定义为b[n/2],其中a,b从0开始标号,除法取下整.给你一个长度为n的序列s.回答Q个这样的询问:s的左端点在[a,b]之间,右端点在[ ...

  6. 【bzoj1858】[Scoi2010]序列操作 线段树区间合并

    题目描述 lxhgww最近收到了一个01序列,序列里面包含了n个数,这些数要么是0,要么是1,现在对于这个序列有五种变换操作和询问操作: 0 a b 把[a, b]区间内的所有数全变成0 1 a b ...

  7. hdu 3308(线段树区间合并)

    LCIS Time Limit: 6000/2000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submis ...

  8. 【BZOJ3638】Cf172 k-Maximum Subsequence Sum 线段树区间合并(模拟费用流)

    [BZOJ3638]Cf172 k-Maximum Subsequence Sum Description 给一列数,要求支持操作: 1.修改某个数的值 2.读入l,r,k,询问在[l,r]内选不相交 ...

  9. 【bzoj3638】Cf172 k-Maximum Subsequence Sum 模拟费用流+线段树区间合并

    题目描述 给一列数,要求支持操作: 1.修改某个数的值 2.读入l,r,k,询问在[l,r]内选不相交的不超过k个子段,最大的和是多少. 输入 The first line contains inte ...

随机推荐

  1. osg::readPixels,glreadPixels截图,保存图片的alpha不对,总是255(1)

    这个函数最近折磨了我很久很久,因为需要用osg截图保存到本地,但是这个图片要具有alpha值,也就是背景的alpha值全为0,但是在公司上用_image->readPixels(448, 28, ...

  2. Spring Security Web应用入门环境搭建

    在使用Spring Security配置Web应用之前,首先要准备一个基于Maven的Spring框架创建的Web应用(Spring MVC不是必须的),本文的内容都是基于这个前提下的. pom.xm ...

  3. 刷屏的海底捞超级APP究竟是怎样与阿里云合作的

    海底捞正式发布了千人千面超级App已有两月,这家餐饮企业总能带给人们不一样的创新能力.谁能想到25年前从四川起家的火锅店,现在门店遍布国内近100座城市,已开门店超400家,海外门店也有50多家,全球 ...

  4. Django项目:CRM(客户关系管理系统)--71--61PerfectCRM实现CRM学生上传作业

    # student_urls.py # ————————60PerfectCRM实现CRM学生上课记录———————— from django.conf.urls import url from bp ...

  5. 深入浅出 Java Concurrency (22): 并发容器 part 7 可阻塞的BlockingQueue (2)[转]

    在上一节中详细分析了LinkedBlockingQueue 的实现原理.实现一个可扩展的队列通常有两种方式:一种方式就像LinkedBlockingQueue一样使用链表,也就是每一个元素带有下一个元 ...

  6. sqoop的数据抽取过程记录

    今天公司抽取了4千万的表大概十几G 用sqoop抽取是30--40分钟 开了两个map.模型是oracle----hdfs(hive).以前只抽过几十万级别,所以千万级别感觉还是spilt做好切分和定 ...

  7. ssm整合:搭建环境

    解决配置中文过滤器后,存入数据库时依旧乱码问题:在web.xml中修改数据库url如下: <property name="jdbcUrl" value="jdbc: ...

  8. 针对老式浏览器(主要是IE6、7、8)的css3-mediaqueries.js自适应布局

    <meta name="viewport" content="width=device-width, initial-scale=1" /> vie ...

  9. css3正方体效果

    <!doctype html> <html> <head> <meta charset="utf-8"> <title> ...

  10. js中的定义变量之①用不用var

    var 是js定义变量的意思. 由于js中的变量是弱类型的,因此js中的所有变量包括number(数字型).string(字符串类型).boolean(布尔类型,true和false)等均通过var关 ...