其他分享
首页 > 其他分享> > 【ZJOI2020】 传统艺能【矩阵快速幂】【DP】

【ZJOI2020】 传统艺能【矩阵快速幂】【DP】

作者:互联网

Description

给定一棵 \([1,n]\) 范围内的广义线段树,\(k\) 次等概率随机选择一个区间 \([l,r]\) 执行区间覆盖操作,问最终 \(tag=1\) 的节点数量的期望。

\(n\le 200000,k\le 10^9\)。

Solution

根据期望的线性性,可以先求出每个节点最终 \(tag=1\) 的概率,加起来即为答案。

考虑每一个区间覆盖对当前节点 \(tag\) 的影响,因为祖先节点可能会下传 \(tag\),因此也要考虑祖先节点的状态。可以将状态分为:

\(3\) 种状态。

不妨设父亲节点 \(f\) 的区间为 \([L,R]\),当前节点 \(p\) 的区间为 \([l,r]\),询问区间为 \([ql,qr]\)。

分类讨论这 \(5\) 种情况,设 \(f_{i,0/1/2}\) 表示考虑了前 \(i\) 次修改,当前状态为 \(0/1/2\) 的概率,不难列出转移方程,然后矩阵快速幂优化即可做到 \(\mathcal O(3^3n\log k)\).

Code

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N=4e5+10,mod=998244353;
int n,k,m[N],tot,L[N],R[N],fa[N],p[N],q[N],g[N],r[N];
inline void dfs(int l,int r,int f){
	int p=++tot;L[p]=l;R[p]=r;fa[p]=f;
	if(L[p]!=R[p]){
		scanf("%d",&m[p]);
		dfs(l,m[p],p);dfs(m[p]+1,r,p);
	}
}
inline int add(int x,int y){return (x+y>=mod)?x+y-mod:x+y;}
inline int dec(int x,int y){return (x-y<0)?x-y+mod:x-y;}
inline int ksm(int x,int y){
	int ret=1;
	for(;y;y>>=1,x=1ll*x*x%mod) if(y&1) ret=1ll*ret*x%mod;
	return ret;
} 
struct matrix{
	int c[3][3];
	inline void init(int d){
		for(int i=0;i<3;++i) for(int j=0;j<3;++j) c[i][j]=i==j?d:0;
	}
	inline matrix operator *(const matrix &A){
		matrix ret;
		for(int i=0;i<3;++i){
			for(int j=0;j<3;++j){
				ll tmp=0;
				for(int k=0;k<3;++k) tmp+=1ll*c[i][k]*A.c[k][j];
				ret.c[i][j]=tmp%mod;
			}
		}
		return ret;
	}
	inline matrix operator ^(int k){
		matrix ret,A=*this;ret.init(1);
		for(;k;k>>=1,A=A*A) if(k&1) ret=ret*A;
		return ret;
	}
};
inline int S(int x){return (1ll*x*(x+1)/2)%mod;}
int main(){
	scanf("%d%d",&n,&k);
	dfs(1,n,0);
	int al=S(n),all=ksm(S(n),mod-2),ans=0;
	for(int i=1;i<=tot;++i){
		int l=L[i],r=R[i],fl=L[fa[i]],fr=R[fa[i]];
		int p1=add(S(fl-1),S(n-fr));//与父亲无交 
		int p2=1ll*fl*(n-fr+1)%mod;//标记祖先 
		int p3=dec(1ll*l*(n-r+1)%mod,p2);//标记自己
		int p4=add(dec(S(l-1),S(fl-1)),dec(S(n-r),S(n-fr)));//下传父亲,不进入自己 
		int p5=dec(al,add(add(p1,p2),add(p3,p4)));//与自己交但不包含 
		matrix A;A.init(0);
		A.c[0][0]=1ll*add(p1,add(p4,p5))*all%mod;A.c[0][1]=1ll*p5*all%mod;A.c[0][2]=1ll*p5*all%mod;
		A.c[1][0]=1ll*p2*all%mod;A.c[1][1]=1ll*add(p1,p2)*all%mod;
		A.c[2][0]=1ll*p3*all%mod;A.c[2][1]=1ll*add(p3,p4)*all%mod;A.c[2][2]=1ll*dec(al,p5)*all%mod;
		A=A^k;
		ans=add(ans,A.c[2][0]);
	}
	printf("%d\n",ans);
	return 0;
}

标签:le,标记,int,ret,艺能,ZJOI2020,节点,DP,mod
来源: https://www.cnblogs.com/tqxboomzero/p/14883244.html