CF755G-PolandBall and Many Other Balls【倍增FFT】
正题
题目链接:https://www.luogu.com.cn/problem/CF755G
题目大意
\(n\)个东西排成一排,每个组可以选择一个单独的物品或者两个连续的物品,一个物品不同同时在两个组里,但是可以不在组里。对于\(i\in[1,k]\)求分成\(i\)组的方案数。
\(1\leq n\leq 10^9,1\leq k<2^{15}\)
解题思路
有三种方法。
第一种是倍增\(FFT\),设\(f_{i,j}\)表示到第\(i\)个物品选了\(j\)组时的方案数,那么设\(F_n(x)=\sum_{i=0}^kf_{n,i}x^i\)。
考虑把这个\(F\)分成两半,然后考虑中间的选不选就是
\]
我们发现如果需要计算\(F_{2^k}\),那么我们就需要维护\(F_{2^{k-1}},F_{2^{k-1}-1},F_{2^{k-1}-2}\)这三个东西。
但是这三个东西也可以用来计算\(F_{2^k-1},F_{2^k-2}\),所以可以维护这三个东西就行倍增。
然后处理的时候同理维护一个\(F_{m}\)和\(F_{m-1}\)就好了。
时间复杂度\(O(n\log^2 n)\),有点卡常
...
第二种方法是直接组合数学推导。将这个序列提出若干段,每一段之间间隔为\(1\),那么只有最末尾的段能够长度为\(2\)的。
\]
瓶颈在于后面的\(\binom{k}{i}\),也就是要求前后没有重复,所以我们可以考虑允许重复的容斥
\]
\]
\]
就可以卷积了,时间复杂度\(O(n\log n)\)
...
第三种方法是特征方程,回到第一个方法的\(F_n(x)\),我们有
\]
\]
这是一个二次项的递推式,过程就不在论述了,用特征方程化简可以得到
\]
然后上全家桶就好了,时间复杂度也是\(O(n\log n)\)
这里的标程用的是第一种方法。
code
#include<cstdio>
#include<cstring>
#include<algorithm>
#define int long long
using namespace std;
const int N=1<<16,P=998244353;
int n,k,m,r[N],f[3][N],t[3][N],g[2][N];
void fm(int &x){x+=x>>31&P;}
int power(int x,int b){
int ans=1;
while(b){
if(b&1)ans=ans*x%P;
x=x*x%P;b>>=1;
}
return ans;
}
void NTT(int *f,int op){
for(int i=0;i<n;i++)
if(i<r[i])swap(f[i],f[r[i]]);
for(int p=2;p<=n;p<<=1){
int tmp=power(3,(P-1)/p),len=p>>1;
if(op==-1)tmp=power(tmp,P-2);
for(int k=0;k<n;k+=p){
int buf=1;
for(int i=k,tt;i<(k|len);i++){
tt=1ll*buf*f[i|len]%P;
fm(f[i|len]=f[i]-tt);
fm(f[i]=f[i]+tt-P);
buf=1ll*buf*tmp%P;
}
}
}
if(op==-1){
int invn=power(n,P-2);
for(int i=0;i<n;i++)
f[i]=1ll*f[i]*invn%P;
}
return;
}
void print(int x)
{if(x>9)print(x/10);putchar(x%10+'0');return;}
signed main()
{
scanf("%d%d",&m,&k);k++;n=1;
while(n<(k*2))n<<=1;
for(int i=0;i<n;i++)
r[i]=(r[i>>1]>>1)|((i&1)?(n>>1):0);
f[0][0]=f[0][1]=f[1][0]=g[0][0]=1;
for(int d=1;d<=m;d<<=1){
if(m&d){
for(int j=0;j<3;j++){
for(int i=0;i<n;i++)
t[j][i]=(i<k)?f[j][i]:0;
NTT(t[j],1);
}
NTT(g[0],1);NTT(g[1],1);
for(int i=0;i<n;i++){
int b0=g[0][i],b1=g[1][i];
g[0][i]=1ll*b0*t[0][i]%P;
g[1][i]=1ll*b0*t[1][i]%P;
t[0][i]=1ll*t[1][i]*b1%P;
t[1][i]=1ll*t[2][i]*b1%P;
}
NTT(g[0],-1);NTT(g[1],-1);
NTT(t[0],-1);NTT(t[1],-1);
for(int i=0;i<k-1;i++)
(g[0][i+1]+=t[0][i])%=P,
(g[1][i+1]+=t[1][i])%=P;
for(int i=k;i<n;i++)g[0][i]=g[1][i]=0;
}
if(d*2>m)break;
for(int j=0;j<3;j++){
for(int i=0;i<n;i++)
t[j][i]=(i<k)?f[j][i]:0;
NTT(t[j],1);
}
for(int i=0;i<n;i++){
f[0][i]=1ll*t[0][i]*t[0][i]%P;
f[1][i]=1ll*t[1][i]*t[0][i]%P;
f[2][i]=1ll*t[1][i]*t[1][i]%P;
t[0][i]=1ll*t[1][i]*t[1][i]%P;
t[1][i]=1ll*t[1][i]*t[2][i]%P;
t[2][i]=1ll*t[2][i]*t[2][i]%P;
}
for(int j=0;j<3;j++)
NTT(f[j],-1),NTT(t[j],-1);
for(int i=0;i<k-1;i++)
(f[0][i+1]+=t[0][i])%P,
(f[1][i+1]+=t[1][i])%P,
(f[2][i+1]+=t[2][i])%P;
for(int i=k;i<n;i++)f[0][i]=f[1][i]=f[2][i]=0;
}
for(int i=1;i<k;i++)
print(g[0][i]),putchar(' ');
return 0;
}
CF755G-PolandBall and Many Other Balls【倍增FFT】的更多相关文章
- 题解-CF755G PolandBall and Many Other Balls
题面 CF755G PolandBall and Many Other Balls 给定 \(n\) 和 \(m\).有一排 \(n\) 个球,求对于每个 \(1\le k\le m\),选出 \(k ...
- CF755G PolandBall and Many Other Balls/soj 57送饮料
题意:长度为n的序列,相邻两个或单独一个可以划分到一个组,每个元素最多处于一个组. 问恰好分割成k(1<=k<=m)段有多少种方案? 标程: #include<bits/stdc++ ...
- CF755G PolandBall and Many Other Balls 题解
从神 Karry 的题单过来的,然后自己瞎 yy 了一个方法,看题解区里没有,便来写一个题解 一个常数和复杂度都很大的题解 令 \(dp_{i,j}\) 为 在 \(i\) 个球中选 \(j\) 组的 ...
- FFT/NTT复习笔记&多项式&生成函数学习笔记Ⅱ
因为垃圾电脑太卡了就重开了一个... 前传:多项式Ⅰ u1s1 我预感还会有Ⅲ 多项式基础操作: 例题: 26. CF438E The Child and Binary Tree 感觉这题作为第一题还 ...
- 多项式 之 快速傅里叶变换(FFT)/数论变换(NTT)/常用套路【入门】
原文链接https://www.cnblogs.com/zhouzhendong/p/Fast-Fourier-Transform.html 多项式 之 快速傅里叶变换(FFT)/数论变换(NTT)/ ...
- CodeForces 553E Kyoya and Train 动态规划 多项式 FFT 分治
原文链接http://www.cnblogs.com/zhouzhendong/p/8847145.html 题目传送门 - CodeForces 553E 题意 一个有$n$个节点$m$条边的有向图 ...
- 快速傅里叶变换FFT / NTT
目录 FFT 系数表示法 点值表示法 复数 DFT(离散傅里叶变换) 单位根的性质 FFT(快速傅里叶变换) IFFT(快速傅里叶逆变换) NTT 阶 原根 扩展知识 FFT 参考blog: 十分简明 ...
- 北京培训记day1
数学什么的....简直是丧心病狂啊好不好 引入:Q1:前n个数中最多能取几个,使得没有一个数是另一个的倍数 答案:(n/2)上取整 p.s.取后n/2个就好了 Q2:在Q1条件下,和最小为多少 答 ...
- NOI前的考试日志
4.14 网络流专项测试 先看T1,不会,看T2,仙人掌???wtf??弃疗.看T3,貌似最可做了,然后开始刚,刚了30min无果,打了50分暴力,然后接着去看T1,把序列差分了一下,推了会式子,发现 ...
随机推荐
- MySQL主从复制与Atlas读写分离
配置主从复制 1. 增加主从配置 # 主库配置文件 server-id = 1 log-bin = /var/lib/mysql/mysql-bin expire_logs_days = 10 ski ...
- 【springcloud】Zuul高级配置(zuul--3)
转自:https://blog.csdn.net/pengjunlee/article/details/87285673 为路由提供HystrixFallback 当Zuul中某一个路由的断路器被断开 ...
- Thread类的常用方法----多线程基础练习
创建多线程程序的第一种方式----继承Thread类 常用API 构造方法 public Thread() :分配一个新的线程对象. public Thread(String name) :分配一个指 ...
- 一招解决下载或下拉GitHub项目速度太慢的问题
相信很多朋友都有过这样的体验,就是从Github上下载或clone别人的项目时特别慢,甚至还会出现链接意外终止的情况,那么今天就来给大家分享一个提速的方法,步骤也非常简单,亲测有效! 首先进入你的目标 ...
- Qt5之正则表达式
字符 描述 \ 将下一个字符标记为一个特殊字符.或一个原义字符.或一个 向后引用.或一个八进制转义符.例如,'n' 匹配字符 "n".'\n' 匹配一个换行符.序列 '\\' 匹配 ...
- 实例_ Java中的代理模式
静态代理 我们定义一个接口,并且使用代理模式,想要做到的就是在调用这个接口的实现类时在此方法上添加功能. public interface HelloInterface { void sayHello ...
- JDK1.8源码(八)——java.lang.ThreadLocal类
https://www.cnblogs.com/xdd666/p/14734047.html ThreadLocal https://www.cnblogs.com/yanfei1819/p/1473 ...
- NOIP模拟「random·string·queen」
T1:random 我又来白剽博客了: 详细证明请看土哥 土哥写的不是很详细,我在这里详细写一下: 首先,对于f[n]的式子: 加一是那一个对的贡献,大C是选其余的几个数,\(2^ ...
- ☕【Java技术指南】「并发编程专题」CompletionService框架基本使用和原理探究(基础篇)
前提概要 在开发过程中在使用多线程进行并行处理一些事情的时候,大部分场景在处理多线程并行执行任务的时候,可以通过List添加Future来获取执行结果,有时候我们是不需要获取任务的执行结果的,方便后面 ...
- Lambda表达式——注重过程的编程思想
一.使用匿名内部类的匿名对象创建线程和Lambda表达式写法 Lambda表达式写法不用去定义一个Runable接口的实现类: 二.方法入参是一个接口或者接口的实现类 三.对某个类的一些对象实例进行排 ...