[gym102220I]Temperature Survey
(为了方便,以下记$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 }
[gym102220I]Temperature Survey的更多相关文章
- The 13th Chinese Northeast Collegiate Programming Contest
题解: solution Code: A. Apple Business #include<cstdio> #include<algorithm> #include<ve ...
- [LeetCode] Rising Temperature 上升温度
Given a Weather table, write a SQL query to find all dates' Ids with higher temperature compared to ...
- SharePoint 2010 Survey的Export to Spreadsheet功能怎么不见了?
背景信息: 最近用户报了一个问题,说他创建的Survey里将结果导出成Excel文件(Export to spreadsheet)的按钮不见了. 原因排查: 正常情况下,这个功能只存在于SharePo ...
- SharePoint Tricks - Survey
1. SharePoint 2010中,在Survey的问题框中输入HTML代码可以用于插入图片或者链接,具体方法为: 1.1 在问题框中输入html, 1.2 在New Form和Edit Form ...
- SharePoint - 添加图片到Survey的某一问题之上
Survey是SharePoint常用功能之一,而曾经被用户多次问到的问题是能否在Survey的某一问题上添加图片,经过查看,SharePoint Survey不提供此方法,只得谷歌之,得一比较懒但又 ...
- 转:关于数据库压缩技术的Survey
原文来自于:http://outofmemory.cn/mysql/database-compression-tech 昨天给团队内的小伙伴做了一个关于数据库压缩技术的Survey,现将其中可以公开的 ...
- BZOJ2276: [Poi2011]Temperature
2276: [Poi2011]Temperature Time Limit: 20 Sec Memory Limit: 32 MBSubmit: 293 Solved: 117[Submit][S ...
- leetcode-Rising Temperature
Given a Weather table, write a SQL query to find all dates' Ids with higher temperature compared to ...
- Application to find the maximum temperature in the weather dataset
import org.apache.hadoop.fs.Path; import org.apache.hadoop.io.IntWritable; import org.apache.hadoop. ...
随机推荐
- Linux基础安全配置(centos7)
1.帐户口令的生存期不长于90天 sed -i.old 's#99999#90#g' /etc/login.defs egrep "90" /etc/login.defs 2.密码 ...
- 密码学系列之:加密货币中的scrypt算法
目录 简介 scrypt算法 scrypt算法详解 scrypt的使用 简介 为了抵御密码破解,科学家们想出了很多种方法,比如对密码进行混淆加盐操作,对密码进行模式变换和组合.但是这些算法逐渐被一些特 ...
- ☕【Java技术指南】「编译器专题」深入分析探究“静态编译器”(JAVA\IDEA\ECJ编译器)是否可以实现代码优化?
技术分析 大家都知道Eclipse已经实现了自己的编译器,命名为 Eclipse编译器for Java (ECJ). ECJ 是 Eclipse Compiler for Java 的缩写,是 Jav ...
- pycharm中设置自己的文件模板
File>>Settings>>Editor>>File and Code Templates 选择文件类型Python Scripts,输入文件模板类型 #!/u ...
- Ajax样例
$.ajax({ url : "newsservlet",//请求地址 dataType : "json",//数据格式 type : "post&q ...
- 【数据结构与算法Python版学习笔记】图——骑士周游问题 深度优先搜索
骑士周游问题 概念 在一个国际象棋棋盘上, 一个棋子"马"(骑士) , 按照"马走日"的规则, 从一个格子出发, 要走遍所有棋盘格恰好一次.把一个这样的走棋序列 ...
- seata整合多数据源
seata整合多数据源 一.背景 二.整合步骤 1.seata server的搭建 2.引入数据源切换组件 3.引入seata组件 4.配置多数据源 5.关闭seata自己默认的数据源代理 6.配置s ...
- Noip模拟72 2021.10.9
T1 出了个大阴间题 真就以为他出了个大阴间题就没写,打个暴力就跑了 数据范围显然摆明是状压 设$f[sta][0/1]$表示在已经选择的集合$sta$中,$A$的最大值是$A$还是$A+1$ 然后按 ...
- 常用Java API: ArrayList(Vector) 和 LinkedList
摘要: 本文主要介绍ArrayList(Vector)和LinkedList的常用方法, 也就是动态数组和链表. ArrayList ArrayList 类可以实现可增长的对象数组. 构造方法 Arr ...
- Python 字符串的encode与decode
python的str,unicode对象的encode和decode方法 python中的str对象其实就是"8-bit string" ,字节字符串,本质上类似java中的byt ...