[关闭]
@xiaoziyao 2021-03-30T23:57:08.000000Z 字数 2711 阅读 997

AT5697 [AGC041F] Histogram Rooks解题报告

解题报告


AT5697 [AGC041F] Histogram Rooks解题报告:

更好的阅读体验

题意

给定一个列的棋盘,每一列高度为,可以放置若干个車(不用考虑車之间会不会影响),需要把整个棋盘覆盖,求方案数。(

分析

容斥毒瘤题,代码分钟,思维小时。

首先看到这一种棋盘的模型(可见P6453 [COCI2008-2009#4] F)果断建出笛卡尔树,然后考虑在上面dp(就是按照最小值分治,把棋盘分成类似下面这样)。

看到这个覆盖满很容易想到容斥,那么很容易第一时间定义为笛卡尔树上结点有至少个格子没有被覆盖(下文称这些点为关键点)的方案数乘上容斥系数

但是这个状态定义的就是,空间会炸,因此我们改变定义:为笛卡尔树上结点一共有列不能放車(注意,这里是不能放車,这些列有可能是没有关键点的)的方案数乘上容斥系数,那么我们只需要关注每一个极长行连续段上是否有关键点。

我们设不能放車的列集合为,不难发现是无法约束关键点的,因此再次容斥,设关键点所在的列集合,容斥系数为,通过容斥简化问题后,我们讨论当前节点的决策:

设当前节点为,区间长度(列数)为,那么:

那么可以发现我们并不关心,而是只关心是否等于了,直接开一维枚举是否等于

具体操作就是类似树形背包,合并左右儿子的信息并进行当前结点的决策:枚举左儿子有列不能放車,右儿子有列不能放車,那么当前结点有列不能放車。设为左子树和右子树均没有贡献满的方案数,为左子树和右子树至少有一个贡献满的方案数,那么经过讨论不难发现:

  1. int p=lc[x],q=rc[x];
  2. for(int i=0;i<=size[p];i++)
  3. for(int j=0;j<=size[q];j++){
  4. int k=i+j;
  5. int a=1ll*f[p][i][0]*f[q][j][0]%mod;
  6. int b=(1ll*f[p][i][0]*f[q][j][1]%mod+1ll*f[p][i][1]*f[q][j][0]%mod+1ll*f[p][i][1]*f[q][j][1]%mod)%mod;
  7. f[x][k][0]=(f[x][k][0]+a)%mod,f[x][k][1]=(f[x][k][1]+b)%mod;
  8. f[x][k+1][0]=(f[x][k+1][0]-a)%mod,f[x][k+1][1]=(f[x][k+1][1]+a)%mod;
  9. }

记得dp完乘上没有讨论的所有点贡献的的次幂,当然求幂的地方可以直接预处理代替快速幂优化掉

复杂度:

代码

  1. #include<stdio.h>
  2. const int maxn=405,mod=998244353;
  3. int n,rt,top,ans;
  4. int a[maxn],stk[maxn],lc[maxn],rc[maxn],f[maxn][maxn][2],bin[maxn],size[maxn],mul[maxn][maxn][2];
  5. void dfs(int x,int last){
  6. if(x==0)
  7. return ;
  8. int p=lc[x],q=rc[x];
  9. dfs(p,a[x]),dfs(q,a[x]);
  10. for(int i=0;i<=size[p];i++)
  11. for(int j=0;j<=size[q];j++){
  12. int k=i+j,a=1ll*f[p][i][0]*f[q][j][0]%mod,b=(1ll*f[p][i][0]*f[q][j][1]%mod+1ll*f[p][i][1]*f[q][j][0]%mod+1ll*f[i][i][1]*f[q][j][1]%mod)%mod;
  13. f[x][k][0]=(f[x][k][0]+a)%mod,f[x][k][1]=(f[x][k][1]+b)%mod;
  14. f[x][k+1][0]=(f[x][k+1][0]-a)%mod,f[x][k+1][1]=(f[x][k+1][1]+a)%mod;
  15. }
  16. size[x]=size[p]+size[q]+1;
  17. for(int i=0;i<=size[x];i++){
  18. f[x][i][0]=1ll*f[x][i][0]*mul[size[x]-i][a[x]-last][0]%mod;
  19. f[x][i][1]=1ll*f[x][i][1]*mul[size[x]-i][a[x]-last][1]%mod;
  20. }
  21. }
  22. int main(){
  23. scanf("%d",&n);
  24. for(int i=1;i<=n;i++)
  25. scanf("%d",&a[i]);
  26. bin[0]=1;
  27. for(int i=1;i<=n;i++)
  28. bin[i]=(bin[i-1]<<1)%mod;
  29. for(int i=0;i<=n;i++){
  30. mul[i][0][0]=mul[i][0][1]=1;
  31. for(int j=1;j<=n;j++){
  32. mul[i][j][0]=1ll*mul[i][j-1][0]*bin[i]%mod;
  33. mul[i][j][1]=1ll*mul[i][j-1][1]*(bin[i]-1)%mod;
  34. }
  35. }
  36. for(int i=1;i<=n;i++){
  37. int k=top;
  38. while(k>0&&a[stk[k]]>a[i])
  39. k--;
  40. if(k>0)
  41. rc[stk[k]]=i;
  42. if(k<top)
  43. lc[i]=stk[k+1];
  44. stk[++k]=i,top=k;
  45. }
  46. rt=stk[1],f[0][0][0]=1,dfs(rt,0);
  47. for(int i=0;i<=n;i++)
  48. ans=(ans+(f[rt][i][0]+f[rt][i][1])%mod)%mod;
  49. printf("%d\n",(ans+mod)%mod);
  50. return 0;
  51. }
添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注