(为了方便,以下记$a_{0}=0,a_{n+1}=n$​​,并将$n$​​加上1)

构造一个$n$行的网格图,从上到下第$i$行有$a_{i}$个格子,格子左对齐

记第$i$行第$j$个格子为$(i,j)$​,格子集合$\{(i,j)\mid i_{1}\le i\le i_{2}$且$j_{1}\le j\le j_{2}\}$为$([i_{1},i_{2}],[j_{1},j_{2}])$

此时,考虑一条从$(1,1)$​​到$(n,a_{n})$​​的路径(只能向下或向右),令$b_{i}$​​为路径中从第$i$​​行到第$i+1$​​行时的格子编号,不难发现这样的路径与合法的$b_{i}$​​一一对应,那么不妨统计路径数

关于路径数,显然可以dp解决,即令$f_{i,j}$表示从$(1,1)$到$(i,j)$的路经数,则$f_{i,j}=f_{i,j-1}+f_{i-1,j}$

使用分治优化dp转移,当分治到区间$[l,r]$​​​时,需要根据$f_{[l,r],a_{l-1}}$​​​的值求出$f_{r,(a_{l-1},a_{r}]}$​​​​​的值

具体的,分治过程如下——

1.令$mid=\lfloor\frac{l+r}{2}\rfloor$​​​​​,递归左区间$[l,mid]$​​​​​,根据$f_{[l,mid],a_{l-1}}$​​​​​的值求出$f_{mid,(a_{l-1},a_{mid}]}$​​​​​的值

2.根据$f_{(mid,r],a_{l-1}}$​​​​​​​和$f_{mid,(a_{l-1},a_{mid}]}$​​​​​​​的值,快速求出$f_{(mid,r],a_{mid}}$​​​​​​​​和$f_{r,(a_{l-1},a_{mid}]}$​​​​​的值

3.递归右区间$(mid,r]$​,根据$f_{(mid,r],a_{mid}}$​​的值求出$f_{r,(a_{mid},a_{r}])}$​的值

(其中,第二步显然可以写成卷积的形式,直接ntt即可)

边界条件:若$l>r$直接退出,若$l=r$令$f_{l,(a_{l-1},a_{r}]}=f_{l,a_{l-1}}$并退出

另外,有一些细节问题:

1.为了避免位置不合法,需要保证$a_{l-1}<a_{l}$,若不满足则不断增加$l$即可

2.关于$f$的存储,只需要用两个数组,分别存储当前每一行/列递归到最右边的一列/行的$f$值即可

总复杂度为$o(n\log^{2}n)$​​,可以通过​

  1 #include<bits/stdc++.h>
2 using namespace std;
3 #define N 200005
4 #define mod 998244353
5 #define ll long long
6 #define vi vector<int>
7 vi v,vl,vu;
8 int t,n,fac[N<<1],inv[N<<1],rev[N<<3],a[N],f[N],ans[N];
9 int c(int n,int m){
10 return (ll)fac[n]*inv[m]%mod*inv[n-m]%mod;
11 }
12 int qpow(int n,int m){
13 int s=n,ans=1;
14 while (m){
15 if (m&1)ans=(ll)ans*s%mod;
16 s=(ll)s*s%mod;
17 m>>=1;
18 }
19 return ans;
20 }
21 void ntt(vi &a,int n,int p){
22 for(int i=0;i<(1<<n);i++)
23 if (i<rev[i])swap(a[i],a[rev[i]]);
24 for(int i=2;i<=(1<<n);i<<=1){
25 int s=qpow(3,(mod-1)/i);
26 if (p)s=qpow(s,mod-2);
27 for(int j=0;j<(1<<n);j+=i)
28 for(int k=0,ss=1;k<(i>>1);k++,ss=(ll)ss*s%mod){
29 int x=a[j+k],y=(ll)a[j+k+(i>>1)]*ss%mod;
30 a[j+k]=(x+y)%mod;
31 a[j+k+(i>>1)]=(x+mod-y)%mod;
32 }
33 }
34 if (p){
35 int s=qpow((1<<n),mod-2);
36 for(int i=0;i<(1<<n);i++)a[i]=(ll)a[i]*s%mod;
37 }
38 }
39 vi mul(vi a,vi b){
40 int n=0,m=a.size()+b.size()-1;
41 while ((1<<n)<m)n++;
42 for(int i=0;i<(1<<n);i++)rev[i]=(rev[i>>1]>>1)+((i&1)*(1<<n)/2);
43 while (a.size()<(1<<n))a.push_back(0);
44 while (b.size()<(1<<n))b.push_back(0);
45 ntt(a,n,0);
46 ntt(b,n,0);
47 for(int i=0;i<(1<<n);i++)a[i]=(ll)a[i]*b[i]%mod;
48 ntt(a,n,1);
49 return a;
50 }
51 void calc(int l,int r){
52 while ((l<=r)&&(a[l-1]==a[l]))l++;
53 if (l>r)return;
54 if (l==r){
55 for(int i=a[l-1]+1;i<=a[r];i++)ans[i]=f[l];
56 return;
57 }
58 int mid=(l+r>>1);
59 calc(l,mid);
60 vl.clear(),vu.clear();
61 for(int i=mid+1;i<=r;i++)vl.push_back(f[i]);
62 for(int i=a[l-1]+1;i<=a[mid];i++)vu.push_back(ans[i]);
63
64 v.clear();
65 for(int i=0;i<r-mid;i++)v.push_back(c(i+a[mid]-a[l-1]-1,i));
66 v=mul(v,vl);
67 for(int i=0;i<r-mid;i++)f[i+mid+1]=v[i];
68
69 v.clear();
70 for(int i=0;i<r-mid+a[mid]-a[l-1]-1;i++)v.push_back(fac[i]);
71 for(int i=0;i<r-mid;i++)vl[i]=(ll)vl[i]*inv[r-mid-1-i]%mod;
72 v=mul(v,vl);
73 for(int i=0;i<a[mid]-a[l-1];i++)ans[i+a[l-1]+1]=(ll)v[i+r-mid-1]*inv[i]%mod;
74
75 v.clear();
76 for(int i=0;i<a[mid]-a[l-1];i++)v.push_back(c(i+r-mid-1,i));
77 v=mul(v,vu);
78 for(int i=0;i<a[mid]-a[l-1];i++)ans[i+a[l-1]+1]=(ans[i+a[l-1]+1]+v[i])%mod;
79
80 v.clear();
81 for(int i=0;i<r-mid+a[mid]-a[l-1]-1;i++)v.push_back(fac[i]);
82 for(int i=0;i<a[mid]-a[l-1];i++)vu[i]=(ll)vu[i]*inv[a[mid]-a[l-1]-1-i]%mod;
83 v=mul(v,vu);
84 for(int i=0;i<r-mid;i++)f[i+mid+1]=(f[i+mid+1]+(ll)v[i+a[mid]-a[l-1]-1]*inv[i])%mod;
85 calc(mid+1,r);
86 }
87 int main(){
88 fac[0]=inv[0]=inv[1]=1;
89 for(int i=1;i<(N<<1);i++)fac[i]=(ll)fac[i-1]*i%mod;
90 for(int i=2;i<(N<<1);i++)inv[i]=(ll)(mod-mod/i)*inv[mod%i]%mod;
91 for(int i=1;i<(N<<1);i++)inv[i]=(ll)inv[i-1]*inv[i]%mod;
92 scanf("%d",&t);
93 while (t--){
94 scanf("%d",&n);
95 for(int i=1;i<=n;i++)scanf("%d",&a[i]);
96 n++,a[n]=n;
97 for(int i=1;i<=n;i++)f[i]=ans[i]=0;
98 f[1]=1;
99 calc(1,n);
100 printf("%d\n",ans[n]);
101 }
102 return 0;
103 }

最新文章

  1. 02 button的练习
  2. 【USACO 3.3】Riding The Fences(欧拉路径)
  3. kafka集群安装
  4. String类和StringBuffer类的区别
  5. Android开发学习笔记:浅谈显示Intent和隐式Intent
  6. rsync+inotify实现远程数据备份
  7. 使用mac 终端利用alias设置快捷命令
  8. 将本地的新建的web Form页面放到服务器提示错误;
  9. 继承下public,protected,private访问权限
  10. Android 模拟器 获得 root权限
  11. openfire+asmack搭建的安卓即时通讯(三) 15.4.9
  12. 数据抓取的艺术(一):Selenium+Phantomjs数据抓取环境配置
  13. HDU 1018 Big Number
  14. iOS开发-21UINavigationController导航控制器初始化 导航控制器栈的push和pop跳转理解
  15. iOS 请求数据 error
  16. OPENFILENAME使用lpstrFilter过滤文件类型
  17. 201521123045 《Java程序设计》第8周学习总结
  18. vue学习笔记(一)——why Vue
  19. vmware centos7 minimal 配置共享文件夹
  20. xcode8 使用Instruments检测定位并解决iOS内存泄露

热门文章

  1. Min_25筛 学习小记
  2. ArrayList-源码分析-自动扩容机制
  3. 微信小程序_快速入门02
  4. 2020.12.14--Codeforces Round #104 (Div.2)补题
  5. Sentinel-Go 源码系列(一)|开篇
  6. C++ cin和while cin
  7. pycharm安装pika提示CondaHTTPError: HTTP 000 CONNECTION FAILED for url &lt;https://repo.anaconda.com&gt;
  8. javascript-jquery选择器
  9. 21.10.14 test
  10. 矩形覆盖 牛客网 剑指Offer