UOJ#401. 【CTSC2018】青蕈领主 分治,FFT
原文链接www.cnblogs.com/zhouzhendong/p/UOJ401.html
题解
首先,对于一个排列,它的连续段一定只有包含关系,没有相交关系。
我们可以据此得到一棵表示连续段的树。
对于一个连续段节点,它有若干儿子。
由于它的每一个儿子都是连续段,所以我们可以将这些儿子各自看作一个数。设节点x的度数为 d[x]。
设 f[x] 表示 L 数组为 1,1,1,...1,L+1 这样的排列个数,那么答案就是 $\prod f[d[x]]$ 。
然后我们得到了一个关于 f[x] 的迷之式子:
$$f[i] = (i-1) \cdot f[i-1] + \sum_{j=2}^{i-2} (j-1) \cdot f[j] \cdot f[i-j]$$
然后我们考虑对这个东西做分治FFT来求。
这个分治 FFT 的大致流程如下:
求解区间 [L,R] 时,如果 L = R,那么分治结束。
否则,设 mid = (L+R)/2,先 solve(L,mid) 。
然后考虑计算 [L,mid] 对 [mid+1,R] 的贡献。
这时需要分类讨论:
若 L = 1,那么直接 FFT
若 L > 1,那么需要计算两部分贡献:
1. $j\in [2,R-L],i-j\in[L,mid]$
2. $i-j\in [2,R-L], j\in[L,mid]$
各自FFT即可。
时间复杂度 $O(n\log^2 n)$ 。
代码
#include <bits/stdc++.h>
#define clr(x) memset(x,0,sizeof x)
#define For(i,a,b) for (int i=a;i<=b;i++)
#define Fod(i,b,a) for (int i=b;i>=a;i--)
#define fi first
#define se second
#define pb(x) push_back(x)
#define mp(x,y) make_pair(x,y)
#define outval(x) printf(#x" = %d\n",x)
#define outtag(x) puts("---------------"#x"---------------")
#define outarr(a,L,R) printf(#a"[%d..%d] = ",L,R);\
For(_x,L,R)printf("%d ",a[_x]);puts("")
using namespace std;
typedef long long LL;
LL read(){
LL x=0,f=0;
char ch=getchar();
while (!isdigit(ch))
f|=ch=='-',ch=getchar();
while (isdigit(ch))
x=(x<<1)+(x<<3)+(ch^48),ch=getchar();
return f?-x:x;
}
const int N=1<<17,mod=998244353;
int Pow(int x,int y){
int ans=1;
for (;y;y>>=1,x=(LL)x*x%mod)
if (y&1)
ans=(LL)ans*x%mod;
return ans;
}
void Add(int &x,int y){
if ((x+=y)>=mod)
x-=mod;
}
void Del(int &x,int y){
if ((x-=y)<0)
x+=mod;
}
int Del(int x){
return x<0?x+mod:x;
}
namespace fft{
int w[N],R[N];
void init(int n){
int d=0;
while ((1<<d)<n)
d++;
For(i,0,n-1)
R[i]=(R[i>>1]>>1)|((i&1)<<(d-1));
w[0]=1,w[1]=Pow(3,(mod-1)/n);
For(i,2,n-1)
w[i]=(LL)w[i-1]*w[1]%mod;
}
void FFT(int *a,int n,int flag){
if (flag<0)
reverse(w+1,w+n);
For(i,0,n-1)
if (i<R[i])
swap(a[i],a[R[i]]);
for (int t=n>>1,d=1;d<n;d<<=1,t>>=1)
for (int i=0;i<n;i+=d<<1)
for (int j=0;j<d;j++){
int tmp=(LL)w[t*j]*a[i+j+d]%mod;
a[i+j+d]=Del(a[i+j]-tmp);
Add(a[i+j],tmp);
}
if (flag<0){
reverse(w+1,w+n);
int inv=Pow(n,mod-2);
For(i,0,n-1)
a[i]=(LL)a[i]*inv%mod;
}
}
}
using fft::FFT;
namespace pre{
int f[N];
int a[N],b[N];
void solve(int L,int R){
if (L==R){
f[L]=((LL)f[L-1]*(L-1)+f[L])%mod;
return;
}
int mid=(L+R)>>1;
solve(L,mid);
if (L>1){
int n=1;
while (n<(mid-L+1)+(R-L))
n<<=1;
fft::init(n);
memset(a,0,n<<2),memset(b,0,n<<2);
For(i,L,mid)
a[i-L]=f[i];
For(i,2,R-L)
b[i]=(LL)(i-1)*f[i]%mod;
FFT(a,n,1),FFT(b,n,1);
For(i,0,n-1)
a[i]=(LL)a[i]*b[i]%mod;
FFT(a,n,-1);
For(i,mid+1,R)
Add(f[i],a[i-L]);
memset(a,0,n<<2),memset(b,0,n<<2);
For(i,L,mid)
a[i-L]=(LL)(i-1)*f[i]%mod;
For(i,2,R-L)
b[i]=f[i];
FFT(a,n,1),FFT(b,n,1);
For(i,0,n-1)
a[i]=(LL)a[i]*b[i]%mod;
FFT(a,n,-1);
For(i,mid+1,R)
Add(f[i],a[i-L]);
}
else {
int n=1;
while (n<mid*2+1)
n<<=1;
memset(a,0,n<<2),memset(b,0,n<<2);
For(i,2,mid)
a[i]=f[i];
For(i,2,mid)
b[i]=(LL)(i-1)*f[i]%mod;
fft::init(n);
FFT(a,n,1),FFT(b,n,1);
For(i,0,n-1)
a[i]=(LL)a[i]*b[i]%mod;
FFT(a,n,-1);
For(i,mid+1,R)
Add(f[i],a[i]);
}
solve(mid+1,R);
}
void prework(int n){
f[0]=1,f[1]=2;
solve(1,n);
// For(i,2,n){
// f[i]=(LL)f[i-1]*(i-1)%mod;
// For(j,2,i-2)
// Add(f[i],(LL)(j-1)*f[j]%mod*f[i-j]%mod);
// }
//1 2 2 4 16 88 600 4800 43680 443296 4949920 60217408 792134528
}
}
int T,n;
int a[N];
void Solve(){
For(i,1,n)
a[i]=read();
if (a[n]!=n)
return (void)(puts("0"));
int ans=1;
For(i,1,n){
int c=0,j;
for (j=i-1;j>i-a[i];j-=a[j])
c++;
if (j<i-a[i])
return (void)(puts("0"));
ans=(LL)ans*pre::f[c]%mod;
}
printf("%d\n",ans);
}
int main(){
pre::prework(50000);
T=read(),n=read();
while (T--)
Solve();
return 0;
}
UOJ#401. 【CTSC2018】青蕈领主 分治,FFT的更多相关文章
- [CTSC2018]青蕈领主
[CTSC2018]青蕈领主 题解 首先,连续段要知道结论: 连续段要么不交,要么包含 所以是一棵树!每个位置的father是后面第一个包含它的 树形DP! 设dp[x],x为根的子树,(设管辖的区间 ...
- 并不对劲的bzoj5342:loj2554:uoj401:p4566: [Ctsc2018]青蕈领主
题目大意 \(T\)(\(T\leq100\))组询问 有\(1\)到\(n\)(\(n\leq50000\))这\(n\)个整数组成的一个排列 定义这个排列的一个子区间是"连续" ...
- Loj #2554. 「CTSC2018」青蕈领主
Loj #2554. 「CTSC2018」青蕈领主 题目描述 "也许,我的生命也已经如同风中残烛了吧."小绿如是说. 小绿同学因为微积分这门课,对"连续"这一概 ...
- uoj#401. 【CTSC2018】青蕈领主(分治FFT)
传送门 话说分治\(FFT\)是个啥子啊--还有题目里那字好像念(蕈xùn) 首先考虑无解的情况:区间相交或者\(L_n\neq n\) 这两个都可以感性理解一下 所以区间之间只会有包含关系,我们把每 ...
- LOJ 2554 「CTSC2018」青蕈领主——结论(思路)+分治FFT
题目:https://loj.ac/problem/2554 一个“连续”的区间必然是一个排列.所有 r 不同的.len 最长的“连续”区间只有包含.相离,不会相交,不然整个是一个“连续”区间. 只有 ...
- bzoj5342 CTSC2018 Day1T3 青蕈领主
首先显然的是,题中所给出的n个区间要么互相包含,要么相离,否则一定不合法. 然后我们可以对于直接包含的关系建出一棵树,于是现在的问题就是给n个节点分配权值,使其去掉最后一个点后不存在非平凡(长度大于1 ...
- 【UOJ#50】【UR #3】链式反应(分治FFT,动态规划)
[UOJ#50][UR #3]链式反应(分治FFT,动态规划) 题面 UOJ 题解 首先把题目意思捋一捋,大概就是有\(n\)个节点的一棵树,父亲的编号大于儿子. 满足一个点的儿子有\(2+c\)个, ...
- 再探快速傅里叶变换(FFT)学习笔记(其三)(循环卷积的Bluestein算法+分治FFT+FFT的优化+任意模数NTT)
再探快速傅里叶变换(FFT)学习笔记(其三)(循环卷积的Bluestein算法+分治FFT+FFT的优化+任意模数NTT) 目录 再探快速傅里叶变换(FFT)学习笔记(其三)(循环卷积的Blueste ...
- BNUOJ 51279[组队活动 Large](cdq分治+FFT)
传送门 大意:ACM校队一共有n名队员,从1到n标号,现在n名队员要组成若干支队伍,每支队伍至多有m名队员,求一共有多少种不同的组队方案.两个组队方案被视为不同的,当且仅当存在至少一名队员在两种方案中 ...
随机推荐
- (八) Docker 部署 mongodb
参考并感谢 官方文档 https://hub.docker.com/_/mongo 下载mongo镜像(不带tag标签则表示下载latest版本) docker pull mongo 启动 mongo ...
- HTML Marquee跑马灯
Marquee是html的标签,所有的主流浏览器都能兼容,用于创建文字滚动. 来介绍下标签的属性 滚动方向 direction <!--滚动方向 direction 4个值 up down le ...
- Centos6.10编译安装php-7.1.12并安装redis模块
1.服务器初始化 yum update -yyum install epel-release -yyum install gcc gcc-c++ wget lsof lrzsz telnet -y 2 ...
- CPN tools 帮助文档资料和实例
1.替代变迁 包含有替代变迁的页面叫做父页,当CPN网使用替代变迁的时候,替代变迁所表达的逻辑必须在某一个位置得到实现,实现替代变迁逻辑页面叫做子页或者子网. 将替代变迁相邻的库所叫做槽库所,也即是在 ...
- 复习整理(一)——HDFS
1.问题 分布式文件系统那么多,为什么hadoop项目中还要开发一个分布式文件系统呢? 2.存储模型 文件线性字节切割成块(Block),具有offset,id 文件与文件的block大小可以不一样 ...
- 【转】Java奇技淫巧-插件化注解处理API(Pluggable Annotation Processing API)
参考资料 JDK6的新特性之六:插入式注解处理API(Pluggable Annotation Processing API) Java Annotation Processing and Creat ...
- Linux系统下文件压缩与打包命令
Linux系统下文件压缩与打包命令 常用的压缩文件拓展名 * .Z * .zip * .gz * .bz2 * .xz * .tar * .tar.gz * .tar.bz2 * .tar.xz 压缩 ...
- 基于Java+Selenium的WebUI自动化测试框架(十三)-----基础页面类BasePage(Excel)
前面,我们讲了如何使用POI进行Excel的“按需读取”.根据前面我们写的BasePageX,我们可以很轻松的写出来基于这个“按需读取”的BasePage. package webui.xUtils; ...
- 小a的排列(牛客)
题目 题意: 一个长度为n的排列.输入n个数 a[ i ],a[ i ] ∈ [1,n],要求找到长度最小的区间 [ l , r ],满足区间[ l , r ]内的数是连续的,且同时包含 数 x 和 ...
- 《少年先疯队》第八次团队作业:Alpha冲刺第二天
前言 第一天冲刺会议 时间:2019.6.15 地点:宿舍 2.1 今日完成任务情况以及遇到的问题. 2.1.1今日完成任务情况 姚玉婷:房间信息管理功能的实现,如房间的显示, 马丽莎 ...