其他分享
首页 > 其他分享> > [CTS2019]氪金手游(容斥+树形背包DP)

[CTS2019]氪金手游(容斥+树形背包DP)

作者:互联网

降智好题。本蒟蒻VP时没想到怎么做被题面迷惑了,只会20分的“好”成绩。简直自闭了。

首先显然度为0的点是白给的,根据等比数列求和公式即可求得。然后考虑这个树如果是一颗外向树,就是每个点先父亲再自己。然后直接DP,令f[i][j]表示子树i内Σw=j的概率,转移时直接用背包转移一发即可。边是正向的直接转移,反向的加上去掉该限制的答案,并减去反向的答案。复杂度显然是O(n2)

#include<bits/stdc++.h>
using namespace std;
const int N=1019,mod=998244353;
int n,cnt,ans,p[N][4],f[N][N*3],g[N*3],sz[N],inv[N*3],hd[N],v[N<<1],nxt[N<<1],w[N<<1];
void add(int&x,int y){x=x+y>=mod?x+y-mod:x+y;}
int qpow(int a,int b)
{
    int ret=1;
    while(b)
    {
        if(b&1)ret=1ll*ret*a%mod;
        a=1ll*a*a%mod,b>>=1;
    }
    return ret;
}
void adde(int x,int y)
{
    v[++cnt]=y,w[cnt]=1,nxt[cnt]=hd[x],hd[x]=cnt;
    v[++cnt]=x,w[cnt]=0,nxt[cnt]=hd[y],hd[y]=cnt;
}
void dfs(int u,int fa)
{
    f[u][0]=1;
    for(int i=hd[u];i;i=nxt[i])
    if(v[i]!=fa)
    {
        dfs(v[i],u);
        for(int j=0;j<=sz[u]+sz[v[i]];j++)g[j]=0;
        for(int j=0;j<=sz[u];j++)
        for(int k=0;k<=sz[v[i]];k++)
        {
            int val=1ll*f[u][j]*f[v[i]][k]%mod;
            if(w[i])add(g[j+k],val);else add(g[j+k],mod-val),add(g[j],val);
        }
        sz[u]+=sz[v[i]];
        for(int j=0;j<=sz[u];j++)f[u][j]=g[j];
    }
    memset(g,0,sizeof g);
    for(int i=0;i<=sz[u];i++)
    for(int j=1;j<=3;j++)
    add(g[i+j],1ll*f[u][i]*p[u][j]%mod*j%mod*inv[i+j]%mod);
    sz[u]+=3;
    for(int i=0;i<=sz[u];i++)f[u][i]=g[i];
}
int main()
{
    scanf("%d",&n);
    for(int i=1,x,y,z,sum;i<=n;i++)
    {
        scanf("%d%d%d",&x,&y,&z);
        sum=qpow(x+y+z,mod-2);
        p[i][1]=1ll*x*sum%mod,p[i][2]=1ll*y*sum%mod,p[i][3]=1ll*z*sum%mod;
    }
    inv[0]=inv[1]=1;for(int i=2;i<=n*3;i++)inv[i]=1ll*(mod-mod/i)*inv[mod%i]%mod;
    for(int i=1,x,y;i<n;i++)scanf("%d%d",&x,&y),adde(x,y);
    dfs(1,0);
    for(int i=0;i<=3*n;i++)add(ans,f[1][i]);
    printf("%d",ans);
}
View Code

 

标签:nxt,cnt,int,容斥,ret,CTS2019,金手游,hd,mod
来源: https://www.cnblogs.com/hfctf0210/p/10901589.html