重構了三次.jpgnode
每次都把這個問題想簡單了.jpgios
果真我仍是太菜了.jpgui
這種題的題解能夠一眼秒掉了,FWT+動態DP簡直是裸的一批...spa
那麼接下來,考慮如何維護信息。code
每一個點維護$4$個信息,分別表示,這條鏈自底向上,自上向底,兩端都在這條鏈的輕兒子裏,和兩端爲鏈頭的方案數。遊戲
這樣的話,正常詢問就沒啥問題了,只須要每次修改和初始化的時候FWT一下,而後最後FWT回來便可。get
而後這樣作的話,由於FWT沒有可減性(無法求逆),因此每次須要將輕兒子用線段樹維護一下,而後每次重建便可string
(是我菜了,FWT之間求每一個對應位置的逆元而後乘起來就行...it
剩下的就是DDP正常操做了...io
#include <cstdio> #include <algorithm> #include <cmath> #include <cstring> #include <cstdlib> #include <queue> #include <iostream> #include <bitset> using namespace std; #define N 30005 #define M 128 #define mod 10007 #define inv2 5004 #define lson l,m,rt<<1 #define rson m+1,r,rt<<1|1 int n,m,Q,a[N]; struct node{int to,next;}e[N<<1]; int head[N],cnt,fa[N],anc[N],son[N],siz[N],idx[N],tims,p[N],sson[N]; // ploy struct Ploy { int a[M],len; void FWT(int opt) { for(int k=2,tmp;k<=m;k<<=1) for(int i=0,t=k>>1;i<m;i+=k) for(int j=i;j<i+t;j++) if(opt==1)tmp=a[j],a[j]=(a[j]+a[j+t])%mod,a[j+t]=(mod+tmp-a[j+t])%mod; else tmp=a[j],a[j]=(a[j]+a[j+t])*inv2%mod,a[j+t]=(mod+tmp-a[j+t])*inv2%mod; } Ploy(){} Ploy(int x){len=1;a[0]=x;FWT(1);} Ploy(int p,int x){memset(a,0,sizeof(a));len=m;a[p]=x;FWT(1);} Ploy(int *b,int m){memcpy(a,b,sizeof(a));len=m;} Ploy operator + (const Ploy &b) const { Ploy c;c.len=max(b.len,len); for(int i=0;i<c.len;i++)c.a[i]=(a[i]+b.a[i])%mod; return c; } Ploy operator - (const Ploy &b) const { Ploy c;c.len=max(b.len,len); for(int i=0;i<c.len;i++)c.a[i]=(mod+a[i]-b.a[i])%mod; return c; } Ploy operator * (const Ploy &b) const { Ploy c;c.len=max(b.len,len); for(int i=0;i<c.len;i++)c.a[i]=a[i]*b.a[i]%mod; return c; } int get(int x){FWT(-1);return a[x];} void print(){FWT(-1);for(int i=0;i<m;i++)printf("%d ",a[i]);puts("");FWT(1);} }f[N],g[N],s[N],h[N],one,tmp,A[N]; // Segment_Tree of all the sons vector<Ploy>Tr[N]; vector<int>lsn[N]; int pos[N]; void build(int x,int l,int r,int rt) { if(l==r){Tr[x][rt]=f[lsn[x][l-1]]+one;pos[lsn[x][l-1]]=rt;return ;} int m=(l+r)>>1;build(x,lson);build(x,rson);Tr[x][rt]=Tr[x][rt<<1]*Tr[x][rt<<1|1]; } // Segment_Tree of DDP struct Segment { Ploy a,b,c,d; Segment(){} Segment(Ploy x,Ploy y){a=b=c=x;d=x+y;} Segment operator + (const Segment &A) const { Segment B; B.a=a*A.a; B.b=b+A.b*a; B.c=A.c+A.a*c; B.d=d+A.d+A.b*c; return B; } }tr[N<<2]; void build(int l,int r,int rt) { if(l==r){tr[rt]=Segment(A[p[l]]*g[p[l]],s[p[l]]);return ;} int m=(l+r)>>1;build(lson);build(rson);tr[rt]=tr[rt<<1]+tr[rt<<1|1]; } Segment query(int L,int R,int l,int r,int rt) { if(L<=l&&r<=R)return tr[rt];int m=(l+r)>>1; if(R<=m)return query(L,R,lson);if(L>m)return query(L,R,rson); return query(L,R,lson)+query(L,R,rson); } void Update(int x,int l,int r,int rt) { if(l==r){tr[rt]=Segment(A[p[l]]*g[p[l]],s[p[l]]);return ;} int m=(l+r)>>1;if(x<=m)Update(x,lson);else Update(x,rson);tr[rt]=tr[rt<<1]+tr[rt<<1|1]; } void Update(int x) { Segment tmp=Segment(); while(x) { Update(idx[x],1,n,1);x=anc[x]; if(fa[x]) { tmp=query(idx[x],idx[sson[x]],1,n,1); s[fa[x]]=s[fa[x]]-h[x]+tmp.d;f[x]=tmp.b; h[x]=tmp.d;int rt=pos[x];Tr[fa[x]][rt]=f[x]+one; for(rt>>=1;rt;rt>>=1)Tr[fa[x]][rt]=Tr[fa[x]][rt<<1]*Tr[fa[x]][rt<<1|1]; g[fa[x]]=Tr[fa[x]][1]; }x=fa[x]; } } // Graph void add(int x,int y){e[cnt]=(node){y,head[x]};head[x]=cnt++;} void dfs1(int x,int from) { fa[x]=from;siz[x]=1;f[x]=A[x]; for(int i=head[x];i!=-1;i=e[i].next) { int to1=e[i].to; if(to1!=from)dfs1(to1,x),f[x]=f[x]*(f[to1]+one),h[x]=h[x]+h[to1],siz[x]+=siz[to1],siz[son[x]]<siz[to1]?son[x]=to1:0; }h[x]=h[x]+f[x]; } void dfs2(int x,int top) { anc[x]=top;idx[x]=++tims;p[tims]=x;sson[x]=x;g[x]=one; if(son[x])dfs2(son[x],top),sson[x]=sson[son[x]]; for(int i=head[x];i!=-1;i=e[i].next) { int to1=e[i].to; if(to1!=son[x]&&to1!=fa[x])dfs2(to1,to1),g[x]=g[x]*(f[to1]+one),s[x]=s[x]+h[to1],lsn[x].push_back(to1); } if(!lsn[x].size())return ; Tr[x].resize(lsn[x].size()*4+5); build(x,1,lsn[x].size(),1); } char opt[10]; int main() { int size = 0x8000000; char*p=(char*)malloc(size) + size; __asm__("movl %0, %%esp\n" :: "r"(p) ); scanf("%d%d",&n,&m);memset(head,-1,sizeof(head)); one.a[0]=1;one.len=1;one.FWT(1); for(int i=1;i<=n;i++)scanf("%d",&a[i]),A[i]=Ploy(a[i],1); for(int i=1,x,y;i<n;i++)scanf("%d%d",&x,&y),add(x,y),add(y,x); dfs1(1,0);dfs2(1,1);build(1,n,1);scanf("%d",&Q); while(Q--) { int x,y;scanf("%s%d",opt,&x); if(opt[0]=='Q')tmp=query(idx[1],idx[sson[1]],1,n,1).d,printf("%d\n",tmp.get(x)); else { scanf("%d",&y);a[x]=y;A[x]=Ploy(a[x],1); Update(x); } } }