P6630 [ZJOI2020] 传统艺能

首先,期望总结点个数等于每个点有标记的期望相加,这是期望的可加性。
那么考虑拆贡献,枚举一个点 \(u\),用矩阵乘法优化 \(DP\) 计算祂的期望。
设状态:
\(f_{i,1}\) 表示操作了 \(i\) 次,\(u\) 祖先没有懒标记,\(u\) 没有懒标记的期望。
\(f_{i,2}\) 表示操作了 \(i\) 次,\(u\) 祖先没有懒标记,\(u\) 有懒标记的期望。
\(f_{i,3}\) 表示操作了 \(i\) 次,\(u\) 祖先有懒标记,\(u\) 没有懒标记的期望。
\(f_{i,4}\) 表示操作了 \(i\) 次,\(u\) 祖先有懒标记,\(u\) 有懒标记的期望。
现在将操作分为几类:
为了方便,称区间 \(A\) 和区间 \(B\) 相交为祂们交集不为空并且前者不包含后者
\(A:\) 操作区间与 \(u\) 父亲的区间不相交。
\(B:\) 操作区间与包含 \(u\) 父亲的区间。
\(C:\) 操作区间与 \(u\) 父亲的区间相交,与 \(u\) 的区间不相交。
\(D:\) 操作区间与 \(u\) 父亲的区间相交,与 \(u\) 的区间相交。
\(E:\) 操作区间与 \(u\) 父亲的区间相交,包含 \(u\) 的区间。
然后转移就用这几种操作区更新,用矩阵乘法优化一下。

代码

\(\mathcal O(nk)\) 暴力

#include<bits/stdc++.h>
using namespace std;
namespace IO{
    template<typename T>
    inline void read(T&x){
        x=0;char c=getchar();bool f=0;
        while(!isdigit(c)) c=='-'?f=1:0,c=getchar();
        while(isdigit(c)) x=x*10+c-'0',c=getchar();
        f?x=-x:0;
    }
    template<typename T>
    inline void write(T x){
        if(x==0){putchar('0');return ;}
        x<0?x=-x,putchar('-'):0;short st[50],top=0;
        while(x) st[++top]=x%10,x/=10;
        while(top) putchar(st[top--]+'0');
    }
    inline void read(char&c){c=getchar();while(isspace(c)) c=getchar();}
    inline void write(char c){putchar(c);}
    inline void read(string&s){s.clear();char c;read(c);while(!isspace(c)&&~c) s+=c,c=getchar();}
    inline void write(string s){for(int i=0,len=s.size();i<len;i++) putchar(s[i]);}
    template<typename T>inline void write(T*x){while(*x) putchar(*(x++));}
    template<typename T,typename...T2> inline void read(T&x,T2&...y){read(x),read(y...);}
    template<typename T,typename...T2> inline void write(const T x,const T2...y){write(x),putchar(' '),write(y...),sizeof...(y)==1?putchar('\n'):0;}
}using namespace IO;
template<int mod>struct Modint{
    int z;
    Modint(){z=0;}
    Modint(int x){x%=mod;z=x<0?x+mod:x;}
    Modint(long long x){x%=mod;z=x<0?x+mod:x;}
    Modint(short x){x%=mod;z=x<0?x+mod:x;}
    Modint(char x){x%=mod;z=x<0?x+mod:x;}
    Modint(bool x){x%=mod;z=x<0?x+mod:x;}
    friend Modint operator+(Modint t,Modint t2){Modint ans;ans.z=(t.z+t2.z)%mod;return ans;}
    friend Modint operator*(Modint t,Modint t2){Modint ans;ans.z=1ll*t.z*t2.z%mod;return ans;}
    friend Modint operator-(Modint t,Modint t2){Modint ans;ans.z=(t.z-t2.z)%mod;return ans;}
    Modint operator<<(const int t)const{Modint ans;ans.z=(z<<t)%mod;return ans;}
    Modint operator>>(const int t)const{Modint ans;ans.z=(z>>t)%mod;return ans;}
    Modint&operator+=(const Modint t){z=(z+t.z)%mod;return *this;}
    Modint&operator*=(const Modint t){z=1ll*z*t.z%mod;return *this;}
    Modint&operator-=(const Modint t){z=(z-t.z)%mod;return *this;}
    Modint&operator<<=(const int t){z=(z<<t)%mod;return *this;}
    Modint&operator>>=(const int t){z=(z>>t)%mod;return *this;}
    friend Modint ksm(Modint a,int b){
        Modint ans=1;
        while(b){if(b&1) ans=ans*a;a=a*a,b>>=1;}
        return ans;
    }
    friend void read(Modint&z){
        int x=0;char c=getchar();bool f=0;
        while(!isdigit(c)) c=='-'?f=1:0,c=getchar();
        while(isdigit(c)) x=(x*10ll+c-'0')%mod,c=getchar();
        f?x=-x:0;
        z.z=x;
    }
    friend void write(Modint x){x.z<0?x.z+=mod:0;write(x.z);}
};
const int mod=998244353,maxn=400010;
#define M Modint<mod>
int n,k,ch[maxn][2],mid[maxn],cnt,rt,fa[maxn],l[maxn],r[maxn];
M f[maxn][10];
void build(int&u,int l,int r){
    u=++cnt;
    ::l[u]=l,::r[u]=r;
    if(l==r) return ;
    read(mid[u]);
    build(ch[u][0],l,mid[u]),build(ch[u][1],mid[u]+1,r);
    fa[ch[u][0]]=fa[ch[u][1]]=u;
}
M calc(int x){return (1ll*x*(x+1)/2)%mod;}
M calc2(int l,int r){return calc(n)-calc(l-1)-calc(n-r);}
signed main(){
    read(n,k);
    build(rt,1,n);
    M ans=ksm(calc(n),mod-2);
    for(int i=2;i<=cnt;i++){
        int l=::l[i],r=::r[i],L=::l[fa[i]],R=::r[fa[i]];
        M A,B=M(L)*(n-R+1),C=calc2(L,R)-calc2(l,r),D=calc2(l,r)-M(l)*(n-r+1),E=M(l)*(n-r+1)-B;
        A=calc(n)-B-C-D-E;
        M inv=ksm(calc(n),mod-2);
        A*=inv,B*=inv,C*=inv,D*=inv,E*=inv;
        f[1][1]=1,f[1][2]=f[1][3]=f[1][4]=0;
        for(int j=2;j<=k+1;j++){
            f[j][1]=f[j-1][1]*(A+C+D)+f[j-1][2]*D+f[j-1][3]*D+f[j-1][4]*D;
            f[j][2]=f[j-1][1]*E+f[j-1][2]*(A+C+E)+f[j-1][3]*(C+E)+f[j-1][4]*(C+E);
            f[j][3]=f[j-1][1]*B+f[j-1][3]*(A+B);
            f[j][4]=f[j-1][2]*B+f[j-1][4]*(A+B);
        }
        ans+=f[k+1][2]+f[k+1][4];
    }
    write(ans);
    return 0;
}

\(\mathcal O(nlog(n))\) 正解

/*
Luogu P6630 [ZJOI2020] 传统艺能
2026-03-18
*/
#include<bits/stdc++.h>
using namespace std;
namespace IO{
    template<typename T>
    inline void read(T&x){
        x=0;char c=getchar();bool f=0;
        while(!isdigit(c)) c=='-'?f=1:0,c=getchar();
        while(isdigit(c)) x=x*10+c-'0',c=getchar();
        f?x=-x:0;
    }
    template<typename T>
    inline void write(T x){
        if(x==0){putchar('0');return ;}
        x<0?x=-x,putchar('-'):0;short st[50],top=0;
        while(x) st[++top]=x%10,x/=10;
        while(top) putchar(st[top--]+'0');
    }
    inline void read(char&c){c=getchar();while(isspace(c)) c=getchar();}
    inline void write(char c){putchar(c);}
    inline void read(string&s){s.clear();char c;read(c);while(!isspace(c)&&~c) s+=c,c=getchar();}
    inline void write(string s){for(int i=0,len=s.size();i<len;i++) putchar(s[i]);}
    template<typename T>inline void write(T*x){while(*x) putchar(*(x++));}
    template<typename T,typename...T2> inline void read(T&x,T2&...y){read(x),read(y...);}
    template<typename T,typename...T2> inline void write(const T x,const T2...y){write(x),putchar(' '),write(y...),sizeof...(y)==1?putchar('\n'):0;}
}using namespace IO;
template<int mod>struct Modint{
    int z;
    Modint(){z=0;}
    Modint(int x){x%=mod;z=x<0?x+mod:x;}
    Modint(long long x){x%=mod;z=x<0?x+mod:x;}
    Modint(short x){x%=mod;z=x<0?x+mod:x;}
    Modint(char x){x%=mod;z=x<0?x+mod:x;}
    Modint(bool x){x%=mod;z=x<0?x+mod:x;}
    friend Modint operator+(Modint t,Modint t2){Modint ans;ans.z=(t.z+t2.z)%mod;return ans;}
    friend Modint operator*(Modint t,Modint t2){Modint ans;ans.z=1ll*t.z*t2.z%mod;return ans;}
    friend Modint operator-(Modint t,Modint t2){Modint ans;ans.z=(t.z-t2.z)%mod;return ans;}
    Modint operator<<(const int t)const{Modint ans;ans.z=(z<<t)%mod;return ans;}
    Modint operator>>(const int t)const{Modint ans;ans.z=(z>>t)%mod;return ans;}
    Modint&operator+=(const Modint t){z=(z+t.z)%mod;return *this;}
    Modint&operator*=(const Modint t){z=1ll*z*t.z%mod;return *this;}
    Modint&operator-=(const Modint t){z=(z-t.z)%mod;return *this;}
    Modint&operator<<=(const int t){z=(z<<t)%mod;return *this;}
    Modint&operator>>=(const int t){z=(z>>t)%mod;return *this;}
    friend Modint ksm(Modint a,int b){
        Modint ans=1;
        while(b){if(b&1) ans=ans*a;a=a*a,b>>=1;}
        return ans;
    }
    friend void read(Modint&z){
        int x=0;char c=getchar();bool f=0;
        while(!isdigit(c)) c=='-'?f=1:0,c=getchar();
        while(isdigit(c)) x=(x*10ll+c-'0')%mod,c=getchar();
        f?x=-x:0;
        z.z=x;
    }
    friend void write(Modint x){x.z<0?x.z+=mod:0;write(x.z);}
};
const int mod=998244353,maxn=400010;
#define M Modint<mod>
template<typename T,int maxn>struct Matrix{
    T a[maxn+10][maxn+10];
    Matrix(){memset(a,0,sizeof(a));}
    Matrix operator*(const Matrix b){
        Matrix c;
        for(int i=1;i<=maxn;i++) for(int k=1;k<=maxn;k++) for(int j=1;j<=maxn;j++) c.a[i][j]+=a[i][k]*b.a[k][j];
        return c;
    }
};
int n,k,ch[maxn][2],mid[maxn],cnt,rt,fa[maxn],l[maxn],r[maxn];
Matrix<M,4>base,f;
void build(int&u,int l,int r){
    u=++cnt;
    ::l[u]=l,::r[u]=r;
    if(l==r) return ;
    read(mid[u]);
    build(ch[u][0],l,mid[u]),build(ch[u][1],mid[u]+1,r);
    fa[ch[u][0]]=fa[ch[u][1]]=u;
}
M calc(int x){return (1ll*x*(x+1)/2)%mod;}
M calc2(int l,int r){return calc(n)-calc(l-1)-calc(n-r);}
signed main(){
    read(n,k);
    build(rt,1,n);
    M ans=ksm(calc(n),mod-2);
    for(int i=2;i<=cnt;i++){
        int l=::l[i],r=::r[i],L=::l[fa[i]],R=::r[fa[i]];
        M A,B=M(L)*(n-R+1),C=calc2(L,R)-calc2(l,r),D=calc2(l,r)-M(l)*(n-r+1),E=M(l)*(n-r+1)-B;
        A=calc(n)-B-C-D-E;
        M inv=ksm(calc(n),mod-2);
        A*=inv,B*=inv,C*=inv,D*=inv,E*=inv;
        memset(&f,0,sizeof(f));
        f.a[1][1]=1;
        base.a[1][1]=A+C+D,base.a[1][2]=D,base.a[1][3]=D,base.a[1][4]=D;
        base.a[2][1]=E,base.a[2][2]=A+C+E,base.a[2][3]=C+E,base.a[2][4]=C+E;
        base.a[3][1]=B,base.a[3][2]=0,base.a[3][3]=A+B,base.a[3][4]=0;
        base.a[4][1]=0,base.a[4][2]=B,base.a[4][3]=0,base.a[4][4]=A+B;
        int c=k;
        while(c){
            if(c&1) f=base*f;
            base=base*base;
            c>>=1;
        }
        ans+=f.a[2][1]+f.a[4][1];
    }
    write(ans);
    return 0;
}
posted @ 2026-03-18 21:31  Link-Cut_Trees  阅读(3)  评论(0)    收藏  举报