洛谷5月月赛T30212 玩游戏 【分治NTT + 多项式求ln】
题目链接
题解
式子很容易推出来,二项式定理展开后对于\(k\)的答案即可化简为如下:
\]
是一个卷积的形式
我们只需对所有\(k\)预处理出\(\sum\limits_{i = 1}^{n} a_i^{k}\),\(b\)也是类似的
月赛时并不会,暴力预处理便滚粗了,,
考虑泰勒展开,有这样一个式子:
\]
我们令\(x = ax\)
则
\]
出现了我们想要的\(a_i^{k}\)
我们只需求出
\]
则\(x^k\)对应的系数就是\(\frac{(-1)^{k - 1}\sum\limits_{i = 1}^{n}a_i^{k}}{k}\)
分治\(NTT\) + 多项式求\(ln\)即可
复杂度\(O(nlog^2n)\)
#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<map>
#define Redge(u) for (int k = h[u],to; k; k = ed[k].nxt)
#define REP(i,n) for (int i = 1; i <= (n); i++)
#define mp(a,b) make_pair<int,int>(a,b)
#define cls(s) memset(s,0,sizeof(s))
#define cp pair<int,int>
#define LL long long int
using namespace std;
const int maxn = 400005,maxm = 100005,INF = 1000000000;
inline int read(){
int out = 0,flag = 1; char c = getchar();
while (c < 48 || c > 57){if (c == '-') flag = -1; c = getchar();}
while (c >= 48 && c <= 57){out = (out << 3) + (out << 1) + c - 48; c = getchar();}
return out * flag;
}
const int G = 3,P = 998244353;
int R[maxn];
inline int qpow(int a,LL b){
int re = 1;
for (; b; b >>= 1,a = 1ll * a * a % P)
if (b & 1) re = 1ll * re * a % P;
return re;
}
void NTT(int* a,int n,int f){
for (int i = 0; i < n; i++) if (i < R[i]) swap(a[i],a[R[i]]);
for (int i = 1; i < n; i <<= 1){
int gn = qpow(G,(P - 1) / (i << 1));
for (int j = 0; j < n; j += (i << 1)){
int g = 1,x,y;
for (int k = 0; k < i; k++,g = 1ll * g * gn % P){
x = a[j + k],y = 1ll * g * a[j + k + i] % P;
a[j + k] = (x + y) % P,a[j + k + i] = ((x - y) % P + P) % P;
}
}
}
if (f == 1) return;
int nv = qpow(n,P - 2); reverse(a + 1,a + n);
for (int i = 0; i < n; i++) a[i] = 1ll * a[i] * nv % P;
}
int n,m,a[maxn],b[maxn],c[maxn],A[maxn],B[maxn],cv[maxn],N;
int fac[maxn],fv[maxn],inv[maxn];
void init(){
fac[0] = fac[1] = fv[0] = fv[1] = inv[0] = inv[1] = 1;
for (int i = 2; i <= 100000; i++){
fac[i] = 1ll * fac[i - 1] * i % P;
inv[i] = 1ll * (P - P / i) * inv[P % i] % P;
fv[i] = 1ll * fv[i - 1] * inv[i] % P;
}
}
int F[30][maxn],deg[maxn],cnt;
void solve(int l,int r){
if (l == r){
deg[++cnt] = 1;
F[cnt][0] = 1; F[cnt][1] = c[l];
return;
}
int mid = l + r >> 1;
solve(l,mid); solve(mid + 1,r);
int n = 1,L = 0,a = cnt - 1,b = cnt,m = deg[a] + deg[b];
while (n <= m) n <<= 1,L++;
for (int i = 1; i < n; i++) R[i] = (R[i >> 1] >> 1) | ((i & 1) << (L - 1));
for (int i = deg[a] + 1; i < n; i++) F[a][i] = 0;
for (int i = deg[b] + 1; i < n; i++) F[b][i] = 0;
NTT(F[a],n,1); NTT(F[b],n,1);
for (int i = 0; i < n; i++) F[a][i] = 1ll * F[a][i] * F[b][i] % P;
NTT(F[a],n,-1);
cnt--;
deg[cnt] = m;
for (int i = m + 1; i < n; i++) F[cnt][i] = 0;
}
void Der(int* a,int n){
for (int i = 0; i < n; i++) a[i] = 1ll * a[i + 1] * (i + 1) % P;
a[n] = 0;
}
void Int(int* a,int n){
for (int i = n + 1; i; i--) a[i] = 1ll * a[i - 1] * inv[i] % P;
a[0] = 0;
}
void Inv(int* a,int* b,int deg){
if (deg == 1){b[0] = qpow(a[0],P - 2); return;}
Inv(a,b,(deg + 1) >> 1);
int n = 1,L = 0;
while (n < (deg << 1)) n <<= 1,L++;
for (int i = 1; i < n; i++) R[i] = (R[i >> 1] >> 1) | ((i & 1) << (L - 1));
for (int i = 0; i < deg; i++) c[i] = a[i];
for (int i = deg; i < n; i++) c[i] = 0;
NTT(c,n,1); NTT(b,n,1);
for (int i = 0; i < n; i++)
b[i] = 1ll * ((2ll - 1ll * b[i] * c[i] % P) % P + P) % P * b[i] % P;
NTT(b,n,-1);
for (int i = deg; i < n; i++) b[i] = 0;
}
void Getln(int* a,int* b,int deg){
Inv(a,cv,deg);
Der(a,deg);
int n = 1,L = 0;
while (n <= (deg << 1)) n <<= 1,L++;
for (int i = 1; i < n; i++) R[i] = (R[i >> 1] >> 1) | ((i & 1) << (L - 1));
for (int i = deg; i < n; i++) a[i] = 0;
NTT(a,n,1); NTT(cv,n,1);
for (int i = 0; i < n; i++) a[i] = 1ll * a[i] * cv[i] % P;
NTT(a,n,-1);
for (int i = 0; i <= deg; i++) b[i] = a[i];
Int(b,deg);
}
int main(){
init();
n = read(); m = read(); int v = qpow(1ll * n * m % P,P - 2);
REP(i,n) a[i] = read();
REP(i,m) b[i] = read();
N = read();
REP(i,n) c[i] = a[i];
solve(1,n);
//REP(i,n) printf("%d ",F[1][i]); puts("");
for (int i = n + 1; i <= N; i++) F[1][i] = 0;
Getln(F[1],A,N);
for (int i = 1; i <= N; i++){
if (!(i & 1)) A[i] = P - A[i];
A[i] = 1ll * A[i] * i % P * fv[i] % P;
}
//REP(i,N) printf("%d ",A[i]); puts("");
A[0] = n;
cls(cv);
REP(i,m) c[i] = b[i]; cnt = 0;
solve(1,m);
for (int i = m + 1; i <= N; i++) F[1][i] = 0;
Getln(F[1],B,N);
for (int i = 1; i <= N; i++){
if (!(i & 1)) B[i] = P - B[i];
B[i] = 1ll * B[i] * i % P * fv[i] % P;
}
//REP(i,N) printf("%d ",B[i]); puts("");
B[0] = m;
int n = 1,L = 0;
while (n <= (N << 1)) n <<= 1,L++;
for (int i = 1; i < n; i++) R[i] = (R[i >> 1] >> 1) | ((i & 1) << (L - 1));
for (int i = N + 1; i < n; i++) A[i] = B[i] = 0;
NTT(A,n,1); NTT(B,n,1);
for (int i = 0; i < n; i++) A[i] = 1ll * A[i] * B[i] % P;
NTT(A,n,-1);
for (int i = 1; i <= N; i++)
printf("%lld\n",1ll * A[i] * fac[i] % P * v % P);
return 0;
}
洛谷5月月赛T30212 玩游戏 【分治NTT + 多项式求ln】的更多相关文章
- 【洛谷5月月赛】玩游戏(NTT,生成函数)
[洛谷5月月赛]玩游戏(NTT,生成函数) 题面 Luogu 题解 看一下要求的是什么东西 \((a_x+b_y)^i\)的期望.期望显然是所有答案和的平均数. 所以求出所有的答案就在乘一个逆元就好了 ...
- 洛谷4月月赛R2
洛谷4月月赛R2 打酱油... A.koishi的数学题 线性筛约数和就可以\(O(N)\)了... #include <iostream> #include <cstdio> ...
- 洛谷3月月赛 R1 Step! ZERO to ONE
洛谷3月月赛 R1 Step! ZERO to ONE 普及组难度 290.25/310滚粗 t1 10分的日语翻译题....太难了不会... t2 真·普及组.略 注意长为1的情况 #include ...
- 【LGR-054】洛谷10月月赛II
[LGR-054]洛谷10月月赛II luogu 成功咕掉Codeforces Round #517的后果就是,我\(\mbox{T4}\)依旧没有写出来.\(\mbox{GG}\) . 浏览器 \( ...
- 【LGR-051】洛谷9月月赛
[LGR-051]洛谷9月月赛 luogu 签到题 description 给出\(K\)和质数\(m\),求最小的\(N\)使得\(111....1\)(\(N\)个\(1\))\(\equiv k ...
- 「LGR-049」洛谷7月月赛 D.Beautiful Pair
「LGR-049」洛谷7月月赛 D.Beautiful Pair 题目大意 : 给出长度为 \(n\) 的序列,求满足 \(i \leq j\) 且 $a_i \times a_j \leq \max ...
- 洛谷9月月赛round2
洛谷9月月赛2 t1 题意:懒得说了 分析:模拟 代码: program flag; var a:..,..]of char; n,i,m,j,x,y,ans,k:longint; begin ass ...
- 「P4996」「洛谷11月月赛」 咕咕咕(数论
题目描述 小 F 是一个能鸽善鹉的同学,他经常把事情拖到最后一天才去做,导致他的某些日子总是非常匆忙. 比如,时间回溯到了 2018 年 11 月 3 日.小 F 望着自己的任务清单: 看 iG 夺冠 ...
- 「P4994」「洛谷11月月赛」 终于结束的起点(枚举
题目背景 终于结束的起点终于写下句点终于我们告别终于我们又回到原点…… 一个个 OIer 的竞赛生涯总是从一场 NOIp 开始,大多也在一场 NOIp 中结束,好似一次次轮回在不断上演.如果这次 NO ...
随机推荐
- Maven学习(八)-----Maven依赖机制
Maven依赖机制 在 Maven 依赖机制的帮助下自动下载所有必需的依赖库,并保持版本升级. 案例分析 让我们看一个案例研究,以了解它是如何工作的.假设你想使用 Log4j 作为项目的日志.这里你要 ...
- cocos2dx2.0 帧动画的创建和播放过程 深入分析
一.帧动画的创建过程帧动画的实现有四个不可或缺的类,如下:1.CCSpriteFrame:精灵帧信息.存储帧动画的每一帧的纹理基本信息. class CC_DLL CCSpriteFrame : pu ...
- ddms+adt+jdk的安装及调试开发安卓
_______ ddms+adt+jdk的安装及调试开发安卓 目录 阐述 1 1 jdk安装 1 2 sdk安装 3 3 Eclipse安装 6 4 ADT安装 10 5 Ddms使用 16 ...
- join 中的on和where的区别
数据库在通过连接两张或多张表来返回记录时,都会生成一张中间的临时表, 然后再将这张临时表返回给用户. 在使用left jion时,on和where条件的区别如下: 1.on条件是在生成临时表时使用的条 ...
- 【Pthon入门学习】利用slice实现str的strip函数,类似C#中的string.trim
1.先了解下切片的知识点 切片是str, list,tuple中常用的取部分元素的操作. 例如: L =['北京', '上海', '天津', '深圳', '石家庄'] print(L[0:2]) # ...
- Bin Packing 装箱问题——NPH问题的暴力枚举 状压DP
题目: 给定n(1≤n≤24)个物品,重量分别为wi,装进一些容量为S(S<1e8)的背包,最少需要多少个背包?
- ElasticSearch之CURL操作(有空再去整理)
https://www.cnblogs.com/jing1617/p/8060421.html ElasticSearch之CURL操作 CURL的操作 curl是利用URL语法在命令行方式下工 ...
- 作业 20181120-3 Beta发布
此作业要求参见:https://edu.cnblogs.com/campus/nenu/2018fall/homework/2408 小组介绍 组长:付佳 组员:张俊余 李文涛 孙赛佳 田良 于洋 段 ...
- C语言的问卷调查
1.你对自己的未来有什么规划?做了哪些准备? 未来想当一个网络工程师,为了这个目标我正在努力学习网络.网页及相关的知识. 2.你认为什么是学习?学习有什么用?现在学习动力如何?为什么? 学习就是不断尝 ...
- iOS- 利用AFNetworking(AFN) - 实现文件上传
官方建议AFN的使用方法 1. 定义一个全局的AFHttpClient:包含有 1> baseURL 2> 请求 3> 操作队列 NSOperationQueue 2. 由AFHTT ...