您的位置:首页 > 其它

HDU 5909 Tree Cutting(FWT+树形DP)

2017-08-16 14:51 417 查看
【题目链接】 http://acm.hdu.edu.cn/showproblem.php?pid=5909

【题目大意】

  给出一棵树,其每棵连通子树的价值为其点权的xor和,
  问有多少连通子树的价值为1~m

【题解】

  首先定1为根,转有根树,我们在树的每个节点保存一个权值数组,
  表示与其连通的子树的权值,当一个子树并入其父节点时,
  dp[x][i]=dp[x][i]+dp[x][j]*dp[y][k](j^k==i),我们发现这是一个位运算卷积式子,
  所以树上转移可以用fwt加速。

【代码】

#include <cstdio>
#include <algorithm>
#include <vector>
#include <cstring>
using namespace std;
typedef long long LL;
const int N=1030;
const LL mod=1e9+7;
const LL inv2=(mod+1)/2;
int val
;
LL dp

,tmp
;
vector<int> v
;
void FWT(LL*a,int n){
for(int d=1;d<n;d<<=1)for(int m=d<<1,i=0;i<n;i+=m)for(int j=0;j<d;j++){
LL x=a[i+j],y=a[i+j+d];
a[i+j]=(x+y)%mod,a[i+j+d]=(x-y+mod)%mod;
}
}
void UFWT(LL*a,int n){
for(int d=1;d<n;d<<=1)for(int m=d<<1,i=0;i<n;i+=m)for(int j=0;j<d;j++){
LL x=a[i+j],y=a[i+j+d];
a[i+j]=(x+y)%mod*inv2%mod,a[i+j+d]=(x-y+mod)%mod*inv2%mod;
}
}
int T,n,m,x,y;
LL ans
;
void Cal(LL *a,LL *b){
for(int i=0;i<m;i++)tmp[i]=a[i];
FWT(a,m); FWT(b,m);
for(int i=0;i<m;i++)a[i]=(1ll*a[i]*b[i])%mod;
UFWT(a,m); UFWT(b,m);
for(int i=0;i<m;i++)a[i]=(a[i]+tmp[i])%mod;
}
void DP(int x,int fx){
dp[x][val[x]]=1;
for(auto y:v[x]){
if(y==fx)continue;
DP(y,x);
Cal(dp[x],dp[y]);
}
}
int main(){
scanf("%d",&T);
while(T--){
scanf("%d%d",&n,&m);
for(int i=0;i<=n;i++)v[i].clear();
for(int i=1;i<=n;i++)scanf("%d",&val[i]);
for(int i=1;i<n;i++){
scanf("%d%d",&x,&y);
v[x].push_back(y);
v[y].push_back(x);
}memset(dp,0,sizeof(dp));
memset(ans,0,sizeof(ans));
DP(1,1);
for(int i=0;i<m;i++){
for(int j=1;j<=n;j++)ans[i]=(ans[i]+dp[j][i])%mod;
}for(int i=0;i<m;i++)printf(i<m-1?"%d ":"%d\n",ans[i]);
}return 0;
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: