洛谷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 ...
随机推荐
- GitHub中webhooks的使用
目录 GitHub中的webhooks的配置 对配置的webhooks的进行测试 目前在团队在设计一个应用管理的功能,需要了解到常用代码托管的Webhooks的使用.GitHub中的webhooks首 ...
- .net core 2.1.3可能引发Could not load file or assembly XXXXX的错误
参考文档: https://github.com/aspnet/Home/issues/3503 写在前面 感觉自己现在干的活离开发越来越远了啊,不过也很好,每天能学到不少东西,中文的,英文的,永远也 ...
- day06 再谈编码 and 作业讲解
1. 小数据池,(其他语言又叫常量池) id() 查看变量的内存地址 is和== is 判断内存地址是否一致 == 判断内容是否一致 小数据池的作用: 为了快速的创建字符串对象, 可以减少内存的浪费 ...
- [CF1137]Museums Tour
link \(\text{Description:}\) 一个国家有 \(n\) 个城市,\(m\) 条有向道路组成.在这个国家一个星期有 \(d\) 天,每个城市有一个博物馆. 有个旅行团在城市 \ ...
- 华为笔试——C++特定位数比较
题目:特定位数比较 题目介绍:输入两行数据,第一行为 m 个正整数,以空格隔开:第二行为正整数 n ,且 n<= m:要求对第一行的数字的后三位大小进行排序,输出排行 n 的数字,其中,若不满三 ...
- 使用qemu启动dd制作的img镜像
1. 准备工作 应用场景 在需要单机取证时,需要在不影响对象业务的情况下进行取证或分析,可以使用dd 对目标服务器进行镜像,生成img文件,镜像可以通过winhex进行静态分析.但是想要动态分析服务器 ...
- python中的os.walk
原文出处:https://www.jianshu.com/p/bbad16822eab python中os.walk是一个简单易用的文件.目录遍历器,可以帮助我们高效的处理文件.目录方面的事情. 1. ...
- 欢迎来怼--第三十次Scrum会议
一.小组信息 队名:欢迎来怼 小组成员 队长:田继平 成员:李圆圆,葛美义,王伟东,姜珊,邵朔,阚博文 小组照片 二.开会信息 时间:2017/11/18 17:20~17:53,总计33min. 地 ...
- VS2015做单元测试
1.安装测试插件 2.新建测试用例 这里就用课堂练习找水王 作例子 写一个类waterKing.h和waterKing.cpp //idList.h #pragma once #include< ...
- 按照Right-BICEP要求设计四则运算3程序的单元测试用例
按照Right-BICEP要求: Right——结果是否正确? B——是否所有的边界条件都是正确的? I——能查一下反响关联吗? C——能用其它手段交叉检查一下吗? E——你是否可以强制错误条件发生? ...