http://uoj.ac/problem/428

题解

神仙题。

考虑最后一定是放了一个\(1\),然后把其他位置都删掉了。

再考虑到对于序列中的每个位置都对应了一次操作。

我们可以对于每个放\(1\)的操作,把它这次删掉的位置对应的操作当做它的儿子节点。

这样是一个树形结构,应为最后只能剩下一个\(1\),所以这是一个有根树。

于是我们把问题转化为了有根树计数问题。

我们先设一个\(f[i]\)表示\(i\)个节点的有根树的方案数,\(g[i]\)表示\(i\)个节点的森林的方案数(也可以只有一棵树)。

\(G\)的转移比较简单:

\[G_i=F_i+\sum_{j=1}^{i-2}\binom{i-1}{j-1}F_j*G_{i-j}
\]

转移\(F\)的话需要考虑将多棵树拼接起来。

\[F_i=\sum_{j\subset A}\binom{i-1}{j}G_{i-j-1}+[i-1\subset B]
\]

边界:\(f[1]=g[1]=0\)。

这个东西就可以分治\(FFT\)求了,注意讨论分治区间是否包含左端点。

代码

  1. #include<bits/stdc++.h>
  2. #define N 270009
  3. using namespace std;
  4. typedef long long ll;
  5. const int mod=998244353;
  6. const int _G=3;
  7. const int Gi=332748118;
  8. int rev[N],n,A,B;
  9. ll f[N],g[N],F[N],G[N],a[N],b[N],ni[N],jie[N],fn[N];
  10. inline void MOD(ll &x){x=x>=mod?x-mod:x;}
  11. inline ll rd(){
  12. ll x=0;char c=getchar();bool f=0;
  13. while(!isdigit(c)){if(c=='-')f=1;c=getchar();}
  14. while(isdigit(c)){x=(x<<1)+(x<<3)+(c^48);c=getchar();}
  15. return f?-x:x;
  16. }
  17. inline ll power(ll x,ll y){
  18. ll ans=1;
  19. while(y){
  20. if(y&1)ans=ans*x%mod;
  21. x=x*x%mod;
  22. y>>=1;
  23. }
  24. return ans;
  25. }
  26. inline void NTT(ll *a,int l,int tag){
  27. for(int i=1;i<l;++i)if(i>rev[i])swap(a[i],a[rev[i]]);
  28. for(int i=1;i<l;i<<=1){
  29. ll wn=power(tag?_G:Gi,(mod-1)/(i<<1));
  30. for(int j=0;j<l;j+=(i<<1)){
  31. ll w=1;
  32. for(int k=0;k<i;++k,w=w*wn%mod){
  33. ll x=a[j+k],y=a[i+j+k]*w%mod;
  34. MOD(a[j+k]=x+y);MOD(a[i+j+k]=x-y+mod);
  35. }
  36. }
  37. }
  38. if(!tag){
  39. ll ny=power(l,mod-2);
  40. for(int i=0;i<l;++i)a[i]=a[i]*ny%mod;
  41. }
  42. }
  43. void CDQ(int l,int r){
  44. if(l==r){
  45. if(l==1){
  46. f[l]=g[l]=0;
  47. return;
  48. }
  49. g[l]=g[l]*jie[l-1]%mod;
  50. f[l]=f[l]*jie[l-1]%mod;
  51. if(b[l-1])MOD(f[l]+=1);
  52. MOD(g[l]+=f[l]);
  53. g[l]=g[l]*ni[l]%mod;
  54. fn[l]=f[l]*ni[l-1]%mod;
  55. return;
  56. }
  57. int mid=(l+r)>>1;
  58. CDQ(l,mid);
  59. int len=1,L=0;
  60. while(len<=(r-l+1+mid-l+1))len<<=1,L++;
  61. for(int i=1;i<len;++i)rev[i]=rev[i>>1]>>1|((i&1)<<(L-1));
  62. for(int i=0;i<len;++i)F[i]=a[i];
  63. for(int i=l;i<=mid;++i)G[i-l+1]=g[i];
  64. NTT(F,len,1);NTT(G,len,1);
  65. for(int i=0;i<len;++i)F[i]=F[i]*G[i]%mod;
  66. NTT(F,len,0);
  67. for(int i=mid+1;i<=r;++i)MOD(f[i]+=F[i-l]);
  68. for(int i=0;i<len;++i)F[i]=G[i]=0;
  69. if(l==1){
  70. len=1;L=0;int x=(mid-l+1)*2;
  71. while(len<=x)len<<=1,L++;
  72. for(int i=1;i<len;++i)rev[i]=rev[i>>1]>>1|((i&1)<<(L-1));
  73. for(int i=l;i<=mid;++i)G[i-l]=g[i];
  74. for(int i=0;i<r-l&&i<=mid;++i)F[i]=fn[i];
  75. NTT(F,len,1);NTT(G,len,1);
  76. for(int i=0;i<len;++i)F[i]=F[i]*G[i]%mod;
  77. NTT(F,len,0);
  78. for(int i=mid+1;i<=r;++i)MOD(g[i]+=F[i-l]);
  79. for(int i=0;i<len;++i)F[i]=G[i]=0;
  80. }
  81. else{
  82. len=1;L=0;int x=mid-l+1+r-l+1;
  83. while(len<=x)len<<=1,L++;
  84. for(int i=1;i<len;++i)rev[i]=rev[i>>1]>>1|((i&1)<<(L-1));
  85. for(int i=l;i<=mid;++i)F[i-l]=fn[i];
  86. for(int i=0;i<=r-l&&i<=mid;++i)G[i]=g[i];
  87. NTT(F,len,1);NTT(G,len,1);
  88. for(int i=0;i<len;++i)F[i]=F[i]*G[i]%mod;
  89. NTT(F,len,0);
  90. for(int i=mid+1;i<=r;++i)MOD(g[i]+=F[i-l]);
  91. for(int i=0;i<len;++i)F[i]=G[i]=0;
  92. for(int i=l;i<=mid;++i)G[i-l]=g[i];
  93. for(int i=0;i<=r-l&&i<=mid;++i)F[i]=fn[i];
  94. NTT(F,len,1);NTT(G,len,1);
  95. for(int i=0;i<len;++i)F[i]=F[i]*G[i]%mod;
  96. NTT(F,len,0);
  97. for(int i=mid+1;i<=r;++i)MOD(g[i]+=F[i-l]);
  98. for(int i=0;i<len;++i)F[i]=G[i]=0;
  99. }
  100. CDQ(mid+1,r);
  101. }
  102. int main(){
  103. n=rd();A=rd();B=rd();
  104. jie[0]=1;
  105. for(int i=1;i<=n;++i)jie[i]=jie[i-1]*i%mod;
  106. ni[n]=power(jie[n],mod-2);
  107. for(int i=n-1;i>=0;--i)ni[i]=ni[i+1]*(i+1)%mod;
  108. for(int i=1;i<=A;++i){int x=rd();a[x]=1;}
  109. for(int i=0;i<=n;++i)a[i]=a[i]*ni[i];
  110. for(int i=1;i<=B;++i){int x=rd();b[x]=1;}
  111. if(n==1){
  112. puts("1");
  113. return 0;
  114. }
  115. CDQ(1,n);
  116. cout<<f[n];
  117. return 0;
  118. }

UOJ428. 【集训队作业2018】普通的计数题的更多相关文章

  1. UOJ#428. 【集训队作业2018】普通的计数题

    #428. [集训队作业2018]普通的计数题 模型转化好题 所以变成统计有标号合法的树的个数. 合法限制: 1.根标号比子树都大 2.如果儿子全是叶子,数量B中有 3.如果存在一个儿子不是叶子,数量 ...

  2. 【UOJ#450】【集训队作业2018】复读机(生成函数,单位根反演)

    [UOJ#450][集训队作业2018]复读机(生成函数,单位根反演) 题面 UOJ 题解 似乎是\(\mbox{Anson}\)爷的题. \(d=1\)的时候,随便怎么都行,答案就是\(k^n\). ...

  3. 【UOJ#422】【集训队作业2018】小Z的礼物(min-max容斥,轮廓线dp)

    [UOJ#422][集训队作业2018]小Z的礼物(min-max容斥,轮廓线dp) 题面 UOJ 题解 毒瘤xzy,怎么能搬这种题当做WC模拟题QwQ 一开始开错题了,根本就不会做. 后来发现是每次 ...

  4. [集训队作业2018]蜀道难——TopTree+贪心+树链剖分+链分治+树形DP

    题目链接: [集训队作业2018]蜀道难 题目大意:给出一棵$n$个节点的树,要求给每个点赋一个$1\sim n$之内的权值使所有点的权值是$1\sim n$的一个排列,定义一条边的权值为两端点权值差 ...

  5. UOJ #449. 【集训队作业2018】喂鸽子

    UOJ #449. [集训队作业2018]喂鸽子 小Z是养鸽子的人.一天,小Z给鸽子们喂玉米吃.一共有n只鸽子,小Z每秒会等概率选择一只鸽子并给他一粒玉米.一只鸽子饱了当且仅当它吃了的玉米粒数量\(≥ ...

  6. [UOJ422][集训队作业2018]小Z的礼物——轮廓线DP+min-max容斥

    题目链接: [集训队作业2018]小Z的礼物 题目要求的就是最后一个喜欢的物品的期望得到时间. 根据$min-max$容斥可以知道$E(max(S))=\sum\limits_{T\subseteq ...

  7. UOJ#418. 【集训队作业2018】三角形

    #418. [集训队作业2018]三角形 和三角形没有关系 只要知道儿子放置的顺序,就可以直接模拟了 记录历史最大值 用一个pair(a,b):之后加上a个,期间最大值为增加b个 合并? A1+A2= ...

  8. 2019.2.25 模拟赛T1【集训队作业2018】小Z的礼物

    T1: [集训队作业2018]小Z的礼物 我们发现我们要求的是覆盖所有集合里的元素的期望时间. 设\(t_{i,j}\)表示第一次覆盖第i行第j列的格子的时间,我们要求的是\(max\{ALL\}\) ...

  9. UOJ#422. 【集训队作业2018】小Z的礼物

    #422. [集训队作业2018]小Z的礼物 min-max容斥 转化为每个集合最早被染色的期望时间 如果有x个选择可以染色,那么期望时间就是((n-1)*m+(m-1)*n))/x 但是x会变,中途 ...

  10. uoj450 【集训队作业2018】复读机(生成函数,单位根反演)

    uoj450 [集训队作业2018]复读机(生成函数,单位根反演) uoj 题解时间 首先直接搞出单个复读机的生成函数 $ \sum\limits_{ i = 0 }^{ k } [ d | i ] ...

随机推荐

  1. java基础笔记(11)

    css 样式的设置主要有选择器+声明{}:声明里又分为属性和值: 注释代码:/*注释语句*/ 内联式:写在元素开始的标签里:例:<p style = "color:red;font-s ...

  2. 02: CI(持续集成)/CD(持续交付/持续部署)

    1.1 持续集成.持续交付 介绍   参考博客:https://www.cnblogs.com/cay83/p/8856231.html 1.传统交付 1. 传统软件的开发与交付的周期都很漫长,从需求 ...

  3. Python 入门之 内置模块 -- time模块

    Python 入门之 内置模块 -- time模块 1.time模块 ​ time翻译过来就是时间,这个模块是与时间相关的模块 import time # 内置模块 -- 标准库 (1)time.ti ...

  4. 背包问题: HDU1114Piggy-Bank

    Piggy-Bank Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total ...

  5. 算法(C#版)动态规划和贪心算法

    https://blog.csdn.net/kouzhuanjing1849/article/details/88954811

  6. 在SQL中存储过程的一般语法

    一般分为十种情况,每种语法各不相同: 1. 创建语法 1 2 3 4 5 6 7 create proc | procedure pro_name    [{@参数数据类型} [=默认值] [outp ...

  7. Ubuntu下通过apache建立虚拟主机

    一个搞前端交互的,总会遇到这样那样的,不需要写代码去解决的问题,怎么搞?答:只能去大海里捞,问题很明确但答案不一定靠谱,因为回答的人不用去考虑你是否会给自己系统搞崩溃. 那么我只能把自己经过验证的答案 ...

  8. 一般软件开发流程和BBS表设计

    项目开发流程 需求分析 架构师+产品经理+开发组组长 和客户公司谈需求之前 ,事先需要想一下这个项目要怎么做 里面的坑点提前想好比较简单的解决方案 在跟客户谈的时候有意识的引导客户朝你已经想好的方案上 ...

  9. vue项目引入外部字体

    1.UI设计图有"华文黑体",担心客户端没有该字体,将"huawen.ttf"字体文件,放入项目中: 2.创建一个font.scss(或font.css)文件: ...

  10. HTML的条件注释及hack技术

    在很多时候,前端的兼容性问题,都很让人头痛!幸运的是,微软从去年声明:从2016年1月12日起,微软将停止为IE8(包括IE8)提供技术支持和安全更新.整个前端圈子都沸腾起来,和今年七月份Adobe宣 ...