记录编号 |
373741 |
评测结果 |
AAAAAAAAAAAAAAAAAAAA |
题目名称 |
[SYZOI Round1]滑稽♂树 |
最终得分 |
100 |
用户昵称 |
Go灬Fire |
是否通过 |
通过 |
代码语言 |
C++ |
运行时间 |
3.741 s |
提交时间 |
2017-02-21 18:30:46 |
内存使用 |
37.77 MiB |
显示代码纯文本
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
#define LL long long
#define Inf 2e9
#define Lowbit(x) (x&-x)
#define ls(x) (a[x].lc)
#define rs(x) (a[x].rc)
const int maxn=35000;
const int Max=10000;
struct Node{
int lc,rc,sum;
}a[maxn*100];
struct Edge{
int next,to;
}e[maxn*2];
int len,head[maxn];
int root[maxn],n,cnt,v[maxn],id[maxn],Last[maxn],pos;
int ra[maxn],rb[maxn],s[maxn];
void Add_edge(int x,int y){
len++;
e[len].to=y;
e[len].next=head[x];
head[x]=len;
}
void Insert(int x,int add,int opt,int & rt,int l,int r){
if(!rt || opt){
a[++cnt]=a[rt];
rt=cnt;
}
a[rt].sum+=add;
if(l==r)return;
int mid=(l+r)>>1;
if(x<=mid)Insert(x,add,opt,ls(rt),l,mid);
else Insert(x,add,opt,rs(rt),mid+1,r);
}
void Bit_Update(int x,int p,int add){
for(int i=x;i<=n;i+=Lowbit(i))Insert(p,add,0,s[i],1,Max);
}
void Dfs(int x,int fa){
id[x]=++pos;
root[pos]=root[pos-1];
Insert(v[x],1,1,root[pos],1,Max);
for(int i=head[x];i;i=e[i].next){
int j=e[i].to;
if(j!=fa)Dfs(j,x);
}
Last[x]=pos;
}
int Query(int k,int pr,int rt,int l,int r){
if(l==r)return l;
int res=a[ls(rt)].sum-a[ls(pr)].sum;
for(int i=1;i<=ra[0];i++)res-=a[ls(ra[i])].sum;
for(int i=1;i<=rb[0];i++)res+=a[ls(rb[i])].sum;
int mid=(l+r)>>1;
if(res>=k){
for(int i=1;i<=ra[0];i++)ra[i]=ls(ra[i]);
for(int i=1;i<=rb[0];i++)rb[i]=ls(rb[i]);
return Query(k,ls(pr),ls(rt),l,mid);
}
else{
for(int i=1;i<=ra[0];i++)ra[i]=rs(ra[i]);
for(int i=1;i<=rb[0];i++)rb[i]=rs(rb[i]);
return Query(k-res,rs(pr),rs(rt),mid+1,r);
}
}
int Get_sum(int s,int t,int rt,int l,int r){
if(s<=l && t>=r)return a[rt].sum;
int mid=(l+r)>>1;
if(t<=mid)return Get_sum(s,t,ls(rt),l,mid);
if(s>mid)return Get_sum(s,t,rs(rt),mid+1,r);
return Get_sum(s,t,ls(rt),l,mid)+Get_sum(s,t,rs(rt),mid+1,r);
}
int Sum(int s,int t,int pr,int rt){
int tot=0;
tot+=Get_sum(s,t,rt,1,Max);
tot-=Get_sum(s,t,pr,1,Max);
for(int i=1;i<=ra[0];i++)tot-=Get_sum(s,t,ra[i],1,Max);
for(int i=1;i<=rb[0];i++)tot+=Get_sum(s,t,rb[i],1,Max);
return tot;
}
void Copy(int l,int r){
ra[0]=rb[0]=0;
for(int i=l-1;i;i-=Lowbit(i))ra[++ra[0]]=s[i];
for(int i=r ;i;i-=Lowbit(i))rb[++rb[0]]=s[i];
}
void Init();
int main(){
freopen("hjtree.in","r",stdin);freopen("hjtree.out","w",stdout);
Init();
return 0;
}
void Init(){
scanf("%d",&n);
for(int i=1;i<=n;i++)scanf("%d",&v[i]);
for(int i=1;i<n;i++){
int x,y;scanf("%d%d",&x,&y);
Add_edge(x,y);Add_edge(y,x);
}
Dfs(1,0);
int Q;scanf("%d",&Q);
while(Q--){
int type,u,x,y;scanf("%d",&type);
int l,r;
if(type==1){
scanf("%d%d",&u,&x);
l=id[u],r=Last[u];
// printf("<<%d %d>>",l,r);
Copy(l,r);
printf("%d\n",Query(x,root[l-1],root[r],1,Max));
}
else if(type==2){
scanf("%d%d%d",&u,&x,&y);
l=id[u],r=Last[u];
Copy(l,r);
printf("%d\n",Sum(x,y,root[l-1],root[r]));
}
else if(type==3){
scanf("%d%d",&u,&x);
Bit_Update(id[u],v[u],-1);
v[u]=x;
Bit_Update(id[u],v[u],1);
}
}
}
/*
5
3 4 6 1 2
1 2
1 3
3 4
3 5
7
1 1 4
2 1 1 5
3 4 5
1 1 4
2 3 3 6
3 5 7
1 3 3
*/