UOJ#424 【集训队作业2018】count
题意
我们定义长度为\(n\),每个数为\(1\sim m\)之间的整数且\(1\sim m\)都至少出现一次的序列为合法序列。再定义\(pos(l,r)\)表示这个序列的区间\([l,r]\)之间的最大值出现的位置(如果有多个取最左端),如果两个序列\(A\),\(B\)的所有\(pos\)值都相同,则\(A\)和\(B\)是同构的。问有多少不同构的合法序列。
\(n,m\leq 100000\)。
题解
首先\(n<m\)显然是无解的。
考虑什么样的序列是同构的,那么我们首先要有一个能方便的表示区间最大值的位置的数据结构,那就是笛卡尔树。显然只要两个序列的笛卡尔树同构,这两个序列就同构。
那么关于\(m\)的限制应该怎么办呢?
由于有多个相同的算在最左边,因此可以发现在笛卡尔树中,每个点的左儿子的键值都小于这个点的键值,右儿子则是小于等于。那么如果这颗笛卡尔树要是合法的,就有一个必要条件:记\(len(x)\)代表从根到节点\(x\)经过的走向左儿子的路径数量(简称为左链长度),那么对于任意的\(x\),有\(len(x)<m\)。
接下来我们证明在\(n\geq m\)时这是一个充分条件。考虑用满足条件的笛卡尔树构造一个合法序列,只需要先将最长的链中每个点赋值为\(m-点的深度\)(根的深度为\(0\)),接下来不断寻找最深的没有赋值的点,将其赋值为没有出现过的数中的最小值即可。剩下的点只需要赋值为\(父亲的值-1\)。易证这样一定可以得到一个合法序列。
于是我们只需要求\(n\)个点,左链长度不超过\(m-1\)的二叉树个数即可。
设\(f_{i,j}\)表示\(j\)个点,左链长度不超过\(i\)的二叉树个数,考虑枚举左子树的大小,于是就有:
\]
平凡情况有\(f_{0,j}=1\)。
那么将上式表达为卷积,就有:
\]
等价于:
\]
直接做似乎很不可做,但是通过这个式子我们可以得到\(f_i\)可以分解为\(\frac{a_i}{b_i}\),其中\(a_i\),\(b_i\)是两个次数界为\(O(i)\)的多项式,那么考虑求\(a_i\),\(b_i\):
\]
\]
于是就得到\(a_i=b_{i-1}\),\(b_i=b_{i-1}-a_{i-1}x\),将转移关系用矩阵来表示就得到:
\]
那么可以通过矩阵快速幂来求\(a_i\),\(b_i\)。但是直接在矩阵中用多项式进行计算会很麻烦,因此不妨考虑用单位根代入求点值,用\(\rm IDFT\)插出多项式即可。复杂度\(O(n\log n)\)。
#include<cstdio>
#include<cstring>
#include<algorithm>
using std::swap;
const int mod=998244353;
inline int add(int a,int b)
{
return (a+=b)>=mod?a-mod:a;
}
inline int sub(int a,int b)
{
return (a-=b)<0?a+mod:a;
}
inline int mul(int a,int b)
{
return (long long)a*b%mod;
}
inline int qpow(int a,int b)
{
int res=1;
for(;b;a=mul(a,a),b>>=1)
if(b&1)
res=mul(res,a);
return res;
}
const int N=1e6+5;
int rev[N];
inline void ntt(int *f,int n,int p)
{
int w,wi,u,t;
register int i,j,k;
for(i=0;i<n;i++)
if(i<(rev[i]=i&1?rev[i^1]|n>>1:rev[i>>1]>>1))
swap(f[i],f[rev[i]]);
for(i=1;wi=qpow(qpow(3,(mod-1)/(i<<1)),p^1?mod-2:1),i<<1<=n;i<<=1)
for(j=0;w=1,j<n;j+=i<<1)
for(k=0;k<i;w=mul(w,wi),k++)
u=f[j+k],t=mul(w,f[j+k+i]),f[j+k]=add(u,t),f[j+k+i]=sub(u,t);
if(!~p)
for(w=qpow(n,mod-2),i=0;i<n;i++)
f[i]=mul(w,f[i]);
return;
}
inline void poly_mul(int *f,int *g,int n)
{
register int i;
memset(f+n,0,sizeof(int)*n);
memset(g+n,0,sizeof(int)*n);
ntt(f,n<<1,1);f==g?void():ntt(g,n<<1,1);
for(i=0;i<n<<1;i++)
f[i]=mul(f[i],g[i]);
ntt(f,n<<1,-1);f==g?void():ntt(g,n<<1,-1);
return;
}
int F[N],G[N];
int _g[N];
inline void poly_inv(int *f,int n)
{
register int i,j;
memset(_g,0,sizeof(int)*n);
_g[0]=qpow(f[0],mod-2);
for(i=1;i<<1<=n;i<<=1)
{
memcpy(F,f,sizeof(int)*(i<<1));
memcpy(G,_g,sizeof(int)*i);
poly_mul(G,G,i);poly_mul(F,G,i<<1);
for(j=0;j<i<<1;j++)
_g[j]=sub(add(_g[j],_g[j]),F[j]);
}
memcpy(f,_g,sizeof(int)*n);
return;
}
int a[2][2],b[2][2],res[2][2];
inline void matrix_qpow(int p)
{
res[0][0]=res[1][1]=1;res[0][1]=res[1][0]=0;
for(;p;p>>=1)
{
if(p&1)
{
b[0][0]=add(mul(res[0][0],a[0][0]),mul(res[0][1],a[1][0]));
b[0][1]=add(mul(res[0][0],a[0][1]),mul(res[0][1],a[1][1]));
b[1][0]=add(mul(res[1][0],a[0][0]),mul(res[1][1],a[1][0]));
b[1][1]=add(mul(res[1][0],a[0][1]),mul(res[1][1],a[1][1]));
memcpy(res,b,sizeof(b));
}
b[0][0]=add(mul(a[0][0],a[0][0]),mul(a[0][1],a[1][0]));
b[0][1]=add(mul(a[0][0],a[0][1]),mul(a[0][1],a[1][1]));
b[1][0]=add(mul(a[1][0],a[0][0]),mul(a[1][1],a[1][0]));
b[1][1]=add(mul(a[1][0],a[0][1]),mul(a[1][1],a[1][1]));
memcpy(a,b,sizeof(b));
}
return;
}
int n,m;
int f[N],g[N];
signed main()
{
int _=1<<17,w=1,wi=qpow(3,(mod-1)/_);
register int i;
scanf("%d%d",&n,&m);
if(n<m)
return puts("0"),0;
for(i=0;i<_;i++)
{
a[0][0]=0;a[0][1]=1;a[1][0]=sub(0,w);a[1][1]=1;
matrix_qpow(m);
f[i]=add(res[0][0],res[0][1]);g[i]=add(res[1][0],res[1][1]);
w=mul(w,wi);
}
ntt(f,_,-1);ntt(g,_,-1);
poly_inv(g,_);
poly_mul(f,g,_);
printf("%d\n",f[n]);
return 0;
}
UOJ#424 【集训队作业2018】count的更多相关文章
- uoj #450[集训队作业2018]复读机
传送门 \(d=1\),那么任何时刻都可以\(k\)个复读机的一种,答案为\(k^n\) \(d>1\),可以枚举某个复读机的复读次数(必须是\(d\)的倍数),然后第\(i\)个复读时间为\( ...
- UOJ 422 [集训队作业2018] 小Z的礼物 min-max容斥 期望 轮廓线dp
LINK:小Z的礼物 太精髓了 我重学了一遍min-max容斥 重写了一遍按位或才写这道题的. 还是期望多少时间可以全部集齐. 相当于求出 \(E(max(S))\)表示最后一个出现的期望时间. 根据 ...
- UOJ #449. 【集训队作业2018】喂鸽子
UOJ #449. [集训队作业2018]喂鸽子 小Z是养鸽子的人.一天,小Z给鸽子们喂玉米吃.一共有n只鸽子,小Z每秒会等概率选择一只鸽子并给他一粒玉米.一只鸽子饱了当且仅当它吃了的玉米粒数量\(≥ ...
- 【UOJ#450】【集训队作业2018】复读机(生成函数,单位根反演)
[UOJ#450][集训队作业2018]复读机(生成函数,单位根反演) 题面 UOJ 题解 似乎是\(\mbox{Anson}\)爷的题. \(d=1\)的时候,随便怎么都行,答案就是\(k^n\). ...
- 【UOJ#422】【集训队作业2018】小Z的礼物(min-max容斥,轮廓线dp)
[UOJ#422][集训队作业2018]小Z的礼物(min-max容斥,轮廓线dp) 题面 UOJ 题解 毒瘤xzy,怎么能搬这种题当做WC模拟题QwQ 一开始开错题了,根本就不会做. 后来发现是每次 ...
- UOJ#418. 【集训队作业2018】三角形
#418. [集训队作业2018]三角形 和三角形没有关系 只要知道儿子放置的顺序,就可以直接模拟了 记录历史最大值 用一个pair(a,b):之后加上a个,期间最大值为增加b个 合并? A1+A2= ...
- UOJ#422. 【集训队作业2018】小Z的礼物
#422. [集训队作业2018]小Z的礼物 min-max容斥 转化为每个集合最早被染色的期望时间 如果有x个选择可以染色,那么期望时间就是((n-1)*m+(m-1)*n))/x 但是x会变,中途 ...
- UOJ#428. 【集训队作业2018】普通的计数题
#428. [集训队作业2018]普通的计数题 模型转化好题 所以变成统计有标号合法的树的个数. 合法限制: 1.根标号比子树都大 2.如果儿子全是叶子,数量B中有 3.如果存在一个儿子不是叶子,数量 ...
- uoj450 【集训队作业2018】复读机(生成函数,单位根反演)
uoj450 [集训队作业2018]复读机(生成函数,单位根反演) uoj 题解时间 首先直接搞出单个复读机的生成函数 $ \sum\limits_{ i = 0 }^{ k } [ d | i ] ...
- [UOJ422][集训队作业2018]小Z的礼物——轮廓线DP+min-max容斥
题目链接: [集训队作业2018]小Z的礼物 题目要求的就是最后一个喜欢的物品的期望得到时间. 根据$min-max$容斥可以知道$E(max(S))=\sum\limits_{T\subseteq ...
随机推荐
- calico 排错记录 apt-get install telnet
1.用kubespray部署一个单节点集群,kubectl get pods -n kube-system,结果: calico-node-7v8wx 1/1 Running 0 2dcalico-n ...
- CAN总线学习系列之二——CAN总线与RS485的比较
CAN总线学习系列之二——CAN总线与RS485的比较 上 一节介绍了一下CAN总线的基本知识,那么有人会问,现在的总线格式很多,CAN相对于其他的总线有什么特点啊?这个问题问的好,所以我想与其它总线 ...
- 矩阵乘法&&矩阵快速幂&&最基本的矩阵模型——斐波那契数列
矩阵,一个神奇又令人崩溃的东西,常常用来优化序列递推 在百度百科中,矩阵的定义: 在数学中,矩阵(Matrix)是一个按照长方阵列排列的复数或实数集合 ,最早来自于方程组的系数及常数所构成的方阵.这一 ...
- 几个不常用的 Web API
1. 设备震动 vibrate Navigator.vibrate() 方法使设备(有震动硬件)产生有频率的震动.若设备不支持震动,该方法将无效.若某震动方式已经在进行中(当该方法调用时),则前一个震 ...
- Package 设计3:数据源的提取和使用暂存
SSIS 设计系列: Package设计1:选择数据类型.暂存数据和并发 Package设计2:增量更新 Package 设计3:数据源的提取和使用暂存 在使用SSIS Package处理海量数据时, ...
- Windows:查看IP地址,IP地址对应的机器名,占用的端口,以及占用该端口的应用程
Windows 服务器系列: Windows:查看IP地址,IP地址对应的机器名,占用的端口,以及占用该端口的应用程 Windows:使用Dos命令管理服务(Services) Windows:任务调 ...
- C# Language Specification 5.0 (翻译)第二章 词法结构
程序 C# 程序(program)由至少一个源文件(source files)组成,其正式称谓为编译单元(compilation units)[1].每个源文件都是有序的 Unicode 字符序列.源 ...
- TDD 与 CI 在 Python 中的实践
社区化产品的长久生存之道可能莫过于对迭代周期的控制.还记得以前采用老土的阶段开发的年代,将软件生命周期分为各个阶段,当到达每个阶段的里程碑则集中所有的资源.人力作全面冲刺.每次到了里程碑的检查点冲过了 ...
- 完爆Facebook/GraphQL,APIJSON全方位对比解析(三)-表关联查询
相关阅读: 完爆Facebook/GraphQL,APIJSON全方位对比解析(一)-基础功能 完爆Facebook/GraphQL,APIJSON全方位对比解析(二)-权限控制 自APIJSON发布 ...
- Unity 角色场景传送功能
传送触发器 using System.Collections;using System.Collections.Generic;using UnityEngine;using UnityEngine. ...