HDU 6900 Residual Polynomial 

题意:

给出一个多项式\(f_1(x) = \sum_{i=0}^na_ix^i\)

对于任意\(i>=2\),满足\(f_i(x) = b_i(f_{i-1}(x))'+c_if_{i-1}(x)\)

要求得到\(f_n(x)\)的各次项系数模\(998244353\)

\(n\le 10^5, 0\leq a_i,b_i,c_i < 998244353\)

题解:

考虑把\(f_1,f_2,\cdots,f_n\)写成\(n\)列,其中\(f_{ij}\)表示\(f_i\)的\(j\)次项的系数:

\[\begin{array}{cccc} f_{1,0} & f_{2,0} & f_{3,0} & \cdots & f_{n0,} \\\ f_{1,1} & f_{2,1} & f_{3,1} & \cdots & f_{n,1}\\\ \vdots & \vdots & \vdots &\ddots & \vdots \\\ f_{1,n} & f_{2,n} & f_{3,n} & \cdots & f_{n,n} \end{array}
\]

其中\(f_{1,i}=a_i\)

考虑类似\(dp\)的状态转移,那么可以发现, 对于\(f_{ij}\)存在两种转移:

\[\begin{cases} f_{i,j}\stackrel{c_{i+1}}{\longrightarrow}f_{i+1,j} & i<n \\\ f_{i,j}\stackrel{j\cdot b_{i+1}}{\longrightarrow} f_{i+1,j-1} & i<n , j>0 \end{cases}
\]

其中箭头表示乘自身然后加到右边

那么如果把转移看作边,可以发现每个状态(除了边界)向右连了一条边,向右上连了一条边

我们来考虑\(f_{1,i}\)对\(f_{n,j}\)的贡献,其中\(i\ge j\),可以发现\(dp\)的转移其实就是一条条从\(f_{1,i}\)到\(f_{n,j}\)的路径,那么显然可以把所有路径单独分开来看,那么\(f_{1,i}\)对\(f_{n,j}\)的贡献为:\(f_{1,i}\cdot \sum (\prod \operatorname{pathvalue})\),也就是从\(f_{1,i}\)到\(f_{n,j}\)的所有可行路径的边权乘积的和

先不考虑第二类转移中\(j\cdot b_{i+1}\)的\(j\),那么对于任何转移\(f_{1,i}\rightarrow f_{n,j}\),其实就是对于每个\(2\le k\le n\),选择\(b_k\)或者\(c_k\),其中\(b_k\)选择\(i-j\)个,\(c_k\)选择\(n-1-(i-j)\)个,乘起来然后再把所有方案加起来

我们令选\(x\)个\(b_k\)和\(n-1-x\)个\(c_k\)的所有方案的和为\(F(x)\),那么\(F(x)\)是可以用分治+\(FFT\)来得到

令\(F(l,r,x)\)表示在区间\([l,r)\)中选\(x\)个\(b_k\)的所有方案的和,那么可以得到\(F(l,r,x)=\sum_{i+j=x}F(l,mid,i)\cdot F(mid,r,j)\),其中\(mid = \lfloor \frac{l+r}2\rfloor\)

再算上之前没考虑的\(j\)的贡献乘积,可以得到\(f_{n,j}=\sum_{i-k=j}F(k)\cdot f_{1,i}\cdot \frac{(i-1)!}{(j-1)!}\)

考虑把数组反向,也就是\(f_{i,j}\)和\(f_{i,n-j}\)互换,那么就可以得到\(f_{n,j} = \sum_{i+k=j}F(k)\cdot f_{1,i}\cdot \frac{j!}{i!} = j!\sum_{i+k=j}F(k)\cdot \frac{f_{1,i}}{i!}\)

那么就可以再做一次\(FFT\)就能得到答案了

view code
//#pragma GCC optimize("O3")
//#pragma comment(linker, "/STACK:1024000000,1024000000")
#include<bits/stdc++.h>
using namespace std;
function<void(void)> ____ = [](){ios_base::sync_with_stdio(false); cin.tie(0); cout.tie(0);};
const int MOD = 998244353;
const int FFTN = 1<<19;
const int MAXN = 2e5+7;
#define poly vector<int>
typedef unsigned long long int ull;
int ksm(int a, int b){
int ret = 1;
while(b){
if(b&1) ret = 1ll * ret * a % MOD;
b >>= 1;
a = 1ll * a * a % MOD;
}
return ret;
}
namespace FFT{
int w[FFTN+5],W[FFTN+5],R[FFTN+5];
void FFTinit(){
W[0]=1;
W[1]=ksm(3,(MOD-1)/FFTN);
for(int i = 2; i <= FFTN; i++) W[i]=1ll*W[i-1]*W[1]%MOD;
}
int FFTinit(int n){
int L=1;
for (;L<=n;L<<=1);
for(int i = 0; i <= L - 1; i++) R[i]=(R[i>>1]>>1)|((i&1)?(L>>1):0);
return L;
}
int A[FFTN+5],B[FFTN+5];
ull p[FFTN+5];
void DFT(int *a,int n){
for(int i = 0; i < n; i++) p[R[i]]=a[i];
for(int d = 1; d < n; d <<= 1){
int len=FFTN/(d<<1);
for(int i = 0, j = 0; i < d; i++, j += len) w[i]=W[j];
for(int i = 0; i < n; i += (d<<1))
for (int j = 0; j < d; j++){
int y=p[i+j+d]*w[j]%MOD;
p[i+j+d]=p[i+j]+MOD-y;
p[i+j]+=y;
}
if (d==1<<15)
for(int i = 0; i < n; i++) p[i]%=MOD;
}
for(int i = 0; i < n; i++) a[i]=p[i]%MOD;
}
void IDFT(int *a,int n){
for(int i = 0; i < n; i++) p[R[i]]=a[i];
for (int d=1;d<n;d<<=1){
int len=FFTN/(d<<1);
for (int i=0,j=FFTN;i<d;i++,j-=len) w[i]=W[j];
for (int i=0;i<n;i+=(d<<1))
for (int j=0;j<d;j++){
int y=p[i+j+d]*w[j]%MOD;
p[i+j+d]=p[i+j]+MOD-y;
p[i+j]+=y;
}
if (d==1<<15)
for(int i = 0; i < n; i++) p[i]%=MOD;
}
int val=ksm(n,MOD-2);
for(int i = 0; i < n; i++) a[i]=p[i]*val%MOD;
}
poly Mul(const poly &a,const poly &b){
int sza=a.size()-1,szb=b.size()-1;
poly ans(sza+szb+1);
if (sza<=30||szb<=30){
for(int i = 0; i <= sza; i++) for(int j = 0; j <= szb; j++)
ans[i+j]=(ans[i+j]+1ll*a[i]*b[j])%MOD;
return ans;
}
int L=FFTinit(sza+szb);
for(int i = 0; i < L; i++) A[i]=(i<=sza?a[i]:0);
for(int i = 0; i < L; i++) B[i]=(i<=szb?b[i]:0);
DFT(A,L); DFT(B,L);
for(int i = 0; i < L; i++) A[i]=1ll*A[i]*B[i]%MOD;
IDFT(A,L);
for(int i = 0; i <= sza + szb; i++) ans[i]=A[i];
return ans;
}
}
int fac[MAXN], rfac[MAXN], inv[MAXN];
poly divide(int l, int r, vector<int> &B, vector<int> &C){ return l + 1 == r ? poly({C[l],B[l]}) : FFT::Mul(divide(l,(l+r)>>1,B,C), divide((l+r)>>1,r,B,C)); }
void solve(){
int n; scanf("%d",&n);
vector<int> A(n+1), B(n-1), C(n-1);
for(int &x : A) scanf("%d",&x);
for(int &x : B) scanf("%d",&x);
for(int &x : C) scanf("%d",&x);
poly f = divide(0,n-1,B,C);
reverse(A.begin(),A.end());
for(int i = 0; i <= n; i++) A[i] = 1ll * A[i] * fac[n-i] % MOD;
poly W = FFT::Mul(A,f);
for(int i = 0; i <= n; i++) W[i] = 1ll * W[i] * rfac[n-i] % MOD;
for(int i = n; i >= 0; i--) printf("%d%c",W[i]," \n"[!i]);
}
int main(){
fac[0] = rfac[0] = inv[1] = 1;
for(int i = 1; i < MAXN; i++) fac[i] = 1ll * fac[i-1] * i % MOD;
for(int i = 2; i < MAXN; i++) inv[i] = 1ll * (MOD - MOD / i) * inv[MOD % i] % MOD;
for(int i = 1; i < MAXN; i++) rfac[i] = 1ll * rfac[i-1] * inv[i] % MOD;
FFT::FFTinit();
int tt; for(scanf("%d",&tt); tt--; solve());
return 0;
}

HDU 6900 Residual Polynomial【分治 NTT】的更多相关文章

  1. HDU 5552 Bus Routes(2015合肥现场赛A,计数,分治NTT)

    题意  给定n个点,任意两点之间可以不连边也可以连边.如果连边的话可以染上m种颜色. 求最后形成的图,是一个带环连通图的方案数. 首先答案是n个点的图减去n个点能形成的树. n个点能形成的树的方案数比 ...

  2. HDU 6270 Marriage (2017 CCPC 杭州赛区 G题,生成函数 + 容斥 + 分治NTT)

    题目链接  2017 CCPC Hangzhou Problem G 题意描述很清晰. 考虑每个家庭有且仅有$k$对近亲的方案数: $C(a, k) * C(b, k) * k!$ 那么如果在第$1$ ...

  3. HDU 5279 YJC plays Minecraft (分治NTT优化DP)

    题目传送门 题目大意:有$n$个小岛,每个小岛上有$a_{i}$个城市,同一个小岛上的城市互相连接形成一个完全图,第$i$个小岛的第$a_{i}$个城市和第$i+1$个小岛的第$1$个城市连接,特别地 ...

  4. HDU 5322 Hope (分治NTT优化DP)

    题面传送门 题目大意: 假设现在有一个排列,每个数和在它右面第一个比它大的数连一条无向边,会形成很多联通块. 定义一个联通块的权值为:联通块内元素数量的平方. 定义一个排列的权值为:每个联通块的权值之 ...

  5. hdu 5830 FFT + cdq分治

    Shell Necklace Time Limit: 16000/8000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)T ...

  6. hdu 4812 DTree (点分治)

    D Tree Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 102400/102400 K (Java/Others)Total S ...

  7. #565. 「LibreOJ Round #10」mathematican 的二进制(期望 + 分治NTT)

    题面 戳这里,题意简单易懂. 题解 首先我们发现,操作是可以不考虑顺序的,因为每次操作会加一个 \(1\) ,每次进位会减少一个 \(1\) ,我们就可以考虑最后 \(1\) 的个数(也就是最后的和) ...

  8. LOJ2541 PKUWC2018猎人杀(概率期望+容斥原理+生成函数+分治NTT)

    考虑容斥,枚举一个子集S在1号猎人之后死.显然这个概率是w1/(Σwi+w1) (i∈S).于是我们统计出各种子集和的系数即可,造出一堆形如(-xwi+1)的生成函数,分治NTT卷起来就可以了. #i ...

  9. 【BZOJ-3456】城市规划 CDQ分治 + NTT

    题目链接 http://www.lydsy.com/JudgeOnline/problem.php?id=3456 Solution 这个问题可以考虑dp,利用补集思想 N个点的简单图总数量为$2^{ ...

随机推荐

  1. PHP 插件资源

    PHP   jsonRPC  百度云网盘地址  https://pan.baidu.com/s/1itCIhrdd5bPGJMefNUuKvw   提取码 : ax4d PHP Excel 百度云网盘 ...

  2. 死磕以太坊源码分析之state

    死磕以太坊源码分析之state 配合以下代码进行阅读:https://github.com/blockchainGuide/ 希望读者在阅读过程中发现问题可以及时评论哦,大家一起进步. 源码目录 |- ...

  3. ICMP协议概述

    • ICMP是三层协议,和IP.ARP.ICMP同属三层    • IP协议中的6是代表上层的TCP协议,17代表UDP协议,1代表同层的ICMP协议    • ICMP协议主要用来探测       ...

  4. spring cloud config —— git配置管理

    目录 talk is cheep, show your the code Server端 pom.xml server的application.yml 配置文件 测试Server client端 po ...

  5. Windows DHCP最佳实践(四)

    这是Windows DHCP最佳实践和技巧的最终指南. 如果您有任何最佳做法或技巧,请在下面的评论中发布它们. 在本指南(四)中,我将分享以下DHCP最佳实践和技巧. 使用DHCP中继代理 防止恶意D ...

  6. ctfshow—web—web2

    打开靶机,根据提示是SQL注入 打开后看到登录窗口 方法一.手工注入 抓取数据包 开始SQL注入测试 利用万能密码,登录成功 查看回显位置 查询数据库 查询数据库内数据表 如果想整齐一点显示可以添加g ...

  7. 【一天一个知识点系列】- Http之状态码

    状态码 简介 HTTP 状态码负责表示客户端 HTTP 请求的返回结果. 标记服务器端的处理是否正常. 通知出现的错误等工作 作用及类别 作用:状态码告知从服务器端返回的请求结果 状态码的类别 注意: ...

  8. javascript判断浏览器访问,刷新,返回

    话不多说,直接上 if (window.performance.navigation.type === 0/* 正常访问 */) { // 你要干的事 } else if (window.perfor ...

  9. 阿里云VOD(三)

    一.视频播放器 参考文档:https://help.aliyun.com/document_detail/125570.html?spm=a2c4g.11186623.6.1083.1c53448bl ...

  10. 接口新建学习---HTTP请求默认值

    一.HTTP请求默认值 1.使用场景: 每次访问论坛的地址(服务器名称或IP)是不变的,端口也是不变的,协议也是不变的(http协议),测试的时候需要每个请求都要写一遍,在我们的HTTP请求取样器数量 ...