Description
Rin是个特别好动的少女。
一天Rin来到了一个遥远的都市。这个都市有N个建筑,编号从1到N,其中市中心编号为1,这个都市有M条双向通行的街道,每条街道连接着两个不同的建筑,其中某些街道首尾相连连接成了一个环。Rin通过长时间的走访,已经清楚了这个都市的两个特点:
从市中心出发可以到达所有的建筑物。
任意一条街道最多存在与一个简单环中。令Rin心花怒放的是,每个建筑物都会有拉面售卖。拉面有很多不同的种类,但对于Rin而言只有油腻程度的不同,因此我们把油腻程度相同的拉面看做同一种拉面。由于不同建筑物的拉面的油腻程度可能不同,我们用一个正整数来表示拉面的油腻程度。
要知道,拉面可是Rin的最爱,但是现在到了下班高峰期,都市的交通变得非常的堵塞。Rin只能通过没有被堵死的街道通行,去品尝所在建筑物的拉面。
现在Rin想知道,如果她正在编号为x的建筑物,那么在从市中心到x的所有简单路径经过的街道都被堵死的情况下,Rin可以品尝到的拉面中(注意没有出现的拉面是不能算在里面的):
油腻程度<=y且品尝次数为奇数次的拉面有多少种?
油腻程度<=y且品尝次数为偶数次的拉面有多少种?
Input
第一行两个正整数N,M,含义如题所示。
第二行一共N个正整数,第i个数Ai表示第i个建筑物出售的拉面的油腻程度。
接下来M行,每行两个正整数x,y,表示在建筑物x,y之间有一条双向通行的街道。数据保证1<=x,y<=N。
接下来一行一个正整数Q,表示询问个数。
接下来Q行每行三个非负整数ty,x,y,x表示询问的建筑物编号,y表示油腻程度的限制,ty=0时表示询问偶数,ty=1表示询问奇数。
Output
一共q行,对于每个询问输出一个答案。
Sample Input
5 6
2 1 6 7 7
1 2
1 3
2 4
4 5
4 5
1 3
3
0 3 2
0 3 1
0 1 7
Sample Output
0
0
1
Data Constraint
Solution
仙人掌,有一种套路变成树:详细点这里查看
大概就是弄出环顶,把环上的点连到环顶上。这样和树就完全相同了。
树怎么做呢?
对于每个点,开一个线段树表示每个点的每种拉面偶数次和奇数次的有多少种,维护很简单,线段树合并一下就行了。
Code
#include<cstdio>
#include<cstring>
#include<algorithm>
#define fo(i,a,b) for(int i=a;i<=b;i++)
#define N 101000
using namespace std;
int n,m,a[N],last[N],nxt[N*5],to[N*5],tot=1,fa[N],dfn[N],s[N],K;
int root[N],q[N][3],las[N],xnt[N],dat[N],son[N],sz[N],ans[N],b[N],c[N],d[1010000],qu;
struct node{
int l,r,d1,d2;
}t[N*70];
void putin(int x,int y)
{
nxt[++tot]=last[x];last[x]=tot;to[tot]=y;
}
void putsin(int x,int y)
{
xnt[++tot]=las[x];las[x]=tot;dat[tot]=y;
}
bool cnt(int x,int y){return c[x]<c[y];}
void tarjan(int x,int fat)
{
dfn[x]=++tot;
s[++s[0]]=x;
for(int i=last[x];i;i=nxt[i])
if(i!=fat)
{
int y=to[i];
if(dfn[y]>0)
{
if(dfn[y]<dfn[x]) for(int z=s[0];s[z]!=y;z--) fa[s[z]]=y;
continue;
}
tarjan(y,i^1);
if(fa[y]==0) fa[y]=x;
}
s[0]--;
}
void dg(int x)
{
sz[x]=1;
for(int i=last[x];i;i=nxt[i])
{
int y=to[i];
dg(y);
sz[x]+=sz[y];
if(sz[y]>sz[son[x]]) son[x]=y;
}
}
void add(int &x)
{
if(x>0) return;
x=++tot;
}
void update(int v)
{
t[v].d1=t[t[v].l].d1+t[t[v].r].d1;
t[v].d2=t[t[v].l].d2+t[t[v].r].d2;
}
void merge(int v1,int v2,int i,int j)
{
if(i==j)
{
int x=t[v1].d1^t[v2].d1;
if(x==0) t[v1].d1=1,t[v1].d2=0;
else t[v1].d1=0,t[v1].d2=1;
return;
}
int m=(i+j)/2;
if(t[v1].l==0) t[v1].l=t[v2].l;
else if(t[v2].l>0) merge(t[v1].l,t[v2].l,i,m);
if(t[v1].r==0) t[v1].r=t[v2].r;
else if(t[v2].r>0) merge(t[v1].r,t[v2].r,m+1,j);
update(v1);
}
void change(int v,int i,int j,int x)
{
if(i==j)
{
if(t[v].d1+t[v].d2==0) t[v].d2=1;
else t[v].d1=1-t[v].d1,t[v].d2=1-t[v].d2;
return;
}
int m=(i+j)/2;
if(x<=m) add(t[v].l),change(t[v].l,i,m,x);
else add(t[v].r),change(t[v].r,m+1,j,x);
update(v);
}
int get(int v,int i,int j,int x,int y,int z)
{
if(v==0) return 0;
if(i==x&&j==y)
{
if(z==0) return t[v].d1;
else return t[v].d2;
}
int m=(i+j)/2;
if(y<=m) return get(t[v].l,i,m,x,y,z);
else if(x>m) return get(t[v].r,m+1,j,x,y,z);
else return get(t[v].l,i,m,x,m,z)+get(t[v].r,m+1,j,m+1,y,z);
}
void dfs(int x)
{
if(son[x]>0)
{
dfs(son[x]);
root[x]=root[son[x]];
}
else root[x]=++tot;
for(int i=last[x];i;i=nxt[i])
{
int y=to[i];
if(y==son[x]) continue;
dfs(y);
merge(root[x],root[y],1,K);
}
change(root[x],1,K,a[x]);
for(int i=las[x];i;i=xnt[i])
{
int y=dat[i];
if(q[y][2]>c[b[n]]) d[q[y][2]]=K;
if(d[q[y][2]]>0) ans[y]=get(root[x],1,K,1,d[q[y][2]],q[y][0]);
}
}
int main()
{
freopen("map.in","r",stdin);
freopen("map.out","w",stdout);
scanf("%d%d",&n,&m);
fo(i,1,n) scanf("%d",&c[i]),b[i]=i;
sort(b+1,b+n+1,cnt);
fo(i,1,n)
{
if(c[b[i]]>c[b[i-1]]) K++;
fo(j,c[b[i-1]],c[b[i]]-1) d[j]=K-1;
a[b[i]]=K;
}
d[c[b[n]]]=K;
fo(i,1,m)
{
int x,y;scanf("%d%d",&x,&y);
putin(x,y);putin(y,x);
}
tot=0;
tarjan(1,0);
memset(last,0,sizeof(last));
tot=0;
fo(i,2,n) putin(fa[i],i);
dg(1);
tot=0;
scanf("%d",&qu);
fo(i,1,qu)
{
scanf("%d%d%d",&q[i][0],&q[i][1],&q[i][2]);
putsin(q[i][1],i);
}
tot=0;
dfs(1);
fo(i,1,qu) printf("%d\n",ans[i]);
}