3992: [SDOI2015]序列统计
3992: [SDOI2015]序列统计
分析:
给定一个集和s,求多少个长度为n的序列,满足序列中每个数都属于s,并且所有数的乘积模m等于x。
设$f=\sum\limits_{i=0}^{n - 1} a_i x ^ i \ \ 如果集合中存在i,a_i = 1$
那么答案的生成函数为f自乘n次,这里可以快速幂。这里"乘法"定义是:设多项式a乘多项式b等于c,$\sum\limits_{k=0}^{n - 1} c_k = \sum\limits_{i \times j = k} a_i \times b_j$ 每次“乘法”的复杂度是$m^2$,所以复杂度是$O(m^2logn)$。
考虑优化“乘法”的部分,我们知道多项式乘法利用FFT/NTT可以做到$nlogn$的,看能否转化为多项式乘法,即多项式乘法的定义变为$\sum\limits_{k=0}^{n - 1} c_k = \sum\limits_{i + j = k} a_i \times b_j$。
NTT中,有引入原根的概念,在NTT中,原根的用途相当于单位根。 原根有一个性质:对于mod p下的原根g,$g^1, g^2 \dots g^{p - 1}$互不相同,$g^{p - 1} \equiv 1 \mod p$。而且$g^1, g^2 \dots g^{p - 1}$可以分别表示$1,2 \dots p - 1$。
那么我们对m求出单位根,集合S中出现的每个数,都可以表示为$s_i = g^{t_{s_i}}$
此时对于原来的一个序列y,$\prod y_i = x \mod m$,就变成了$\prod g ^{t_{y_i}} = g^{t_x} \mod m$,即$\sum t_{y_i} = x \mod m - 1$
现在我们求的就是长度为n的序列,序列中每个数都属于集合t,并且所有数的和模(m-1)等于x 如此按照上面的做法,将乘法的定义改为多项式乘法的定义,快速幂+NTT即可复杂度$mlogmlogn$。
注意:多项式乘法中是没有取模的,而这里(i+j)%(m-1),直接将数组加倍,然后NTT完后,大于等于m的加到相应的模m后的位置上即可。
代码:
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<iostream>
#include<cmath>
#include<cctype>
#include<set>
#include<queue>
#include<vector>
#include<map>
using namespace std;
typedef long long LL; inline int read() {
int x=,f=;char ch=getchar();for(;!isdigit(ch);ch=getchar())if(ch=='-')f=-;
for(;isdigit(ch);ch=getchar())x=x*+ch-'';return x*f;
} const int mod = ;
const int N = ;
int vis[N], rev[N], n = , m;
int f[N], g[N], a[N], b[N], inv; int ksm(int a,int b,int p) {
a %= p;
int ans = ;
while (b) {
if (b & ) ans = 1ll * ans * a % p;
a = 1ll * a * a % p;
b >>= ;
}
return ans % p;
}
int Calc(int x) {
if (x == ) return ;
for (int i = ; ; ++i) {
bool flag = ;
for (int j = ; j * j < x; ++j)
if (ksm(i, (x - ) / j, x) == ) { flag = false; break; }
if (flag) return i;
}
}
void NTT(int *a,int n,int ty) {
for (int i = ; i < n; ++i) if (i < rev[i]) swap(a[i], a[rev[i]]);
for (int m = ; m <= n; m <<= ) {
int w1 = ksm(, (mod - ) / m, mod);
if (ty == -) w1 = ksm(w1, mod - , mod);
for (int i = ; i < n; i += m) {
int w = ;
for (int k = ; k < (m >> ); ++k) {
int u = a[i + k], t = 1ll * w * a[i + k + (m >> )] % mod;
a[i + k] = (u + t) % mod;
a[i + k + (m >> )] = (u - t + mod) % mod;
w = 1ll * w * w1 % mod;
}
}
}
}
void mul(int *g,int *f) {
for (int i = ; i < n; ++i) a[i] = g[i] % mod, b[i] = f[i] % mod;
NTT(a, n, );
NTT(b, n, );
for (int i = ; i < n; ++i) a[i] = 1ll * a[i] * b[i] % mod;
NTT(a, n, -);
for (int i = ; i < n; ++i) a[i] = 1ll * a[i] * inv % mod;
for (int i = ; i < m - ; ++i) g[i] = (a[i] + a[i + m - ]) % mod;
}
void solve(int b) {
inv = ksm(n, mod - , mod);
g[] = ;
while (b) {
if (b & ) mul(g, f);
b >>= ;
mul(f, f);
}
}
int main() {
int cnt = read(); m = read(); int x = read(), s = read();
for (int i = ; i <= s; ++i) vis[read()] = ;
int q = Calc(m), pos = -, L = ;
for (int i = , j = ; i < m - ; ++i, j = 1ll * j * q % m) {
if (vis[j]) f[i] = ;
if (j == x) pos = i;
}
int M = (m - ) * ;
while (n < M) n <<= , L ++;
for (int i = ; i < n; ++i) rev[i] = (rev[i >> ] >> ) | ((i & ) << (L - ));
solve(cnt);
if (pos != -) cout << g[pos] % mod;
else cout << ;
return ;
}
3992: [SDOI2015]序列统计的更多相关文章
- BZOJ 3992: [SDOI2015]序列统计 [快速数论变换 生成函数 离散对数]
3992: [SDOI2015]序列统计 Time Limit: 30 Sec Memory Limit: 128 MBSubmit: 1017 Solved: 466[Submit][Statu ...
- [BZOJ 3992][SDOI2015]序列统计
3992: [SDOI2015]序列统计 Time Limit: 30 Sec Memory Limit: 128 MBSubmit: 2275 Solved: 1090[Submit][Stat ...
- BZOJ 3992: [SDOI2015]序列统计 NTT+快速幂
3992: [SDOI2015]序列统计 Time Limit: 30 Sec Memory Limit: 128 MBSubmit: 1155 Solved: 532[Submit][Statu ...
- BZOJ 3992: [SDOI2015]序列统计 快速幂+NTT(离散对数下)
3992: [SDOI2015]序列统计 Description 小C有一个集合S,里面的元素都是小于M的非负整数.他用程序编写了一个数列生成器,可以生成一个长度为N的数列,数列中的每个数都属于集合S ...
- 【BZOJ】3992: [SDOI2015]序列统计 NTT+生成函数
[题意]给定一个[0,m-1]范围内的数字集合S,从中选择n个数字(可重复)构成序列.给定x,求序列所有数字乘积%m后为x的序列方案数%1004535809.1<=n<=10^9,3< ...
- bzoj 3992 [SDOI2015]序列统计——NTT(循环卷积&&快速幂)
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3992 有转移次数.模M余数.方案数三个值,一看就是系数的地方放一个值.指数的地方放一个值.做 ...
- bzoj 3992 [SDOI2015] 序列统计 —— NTT (循环卷积+快速幂)
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3992 (学习NTT:https://riteme.github.io/blog/2016-8 ...
- BZOJ.3992.[SDOI2015]序列统计(DP NTT 原根)
题目链接 \(Description\) 给定\(n,m,x\)和集合\(S\).求\(\prod_{i=1}^na_i\equiv x\ (mod\ m)\)的方案数.其中\(a_i\in S\). ...
- bzoj 3992: [SDOI2015]序列统计【原根+生成函数+NTT+快速幂】
还是没有理解透原根--题目提示其实挺明显的,M是质数,然后1<=x<=M-1 这种计数就容易想到生成函数,但是生成函数是加法,而这里是乘法,所以要想办法变成加法 首先因为0和任何数乘都是0 ...
随机推荐
- 一次存储链路抖动因I/O timeout不同在AIX和HPUX上的不同表现(转)
去年一个故障案例经过时间的沉淀问题没在发生今天有时间简单的总结一下,当时正时午睡时分,突然告警4库8个实例同时不可用,这么大面积的故障多数是有共性的关连,当时查看数据库DB ALERT日志都是I/O错 ...
- show tables from information_schema/performance_schema/sys;
root@localhost:3306.sock [performance_schema]>select version();+------------+| version() |+----- ...
- MySQL · 数据恢复 · undrop-for-innodb
Ref:https://www.aliyun.com/jiaocheng/1109809.html 摘要: 简介 undrop-for-innodb 是针对 innodb 的一套数据恢复工具,可以从 ...
- 什么时候会执行viewDidLoad方法
什么时候会执行viewDidLoad方法 这个博文是为了解释,为何有时候,你给属性赋值,在viewDidLoad方法中却取不到值的原因. 第一种情况,presentViewController会执行被 ...
- [转载]Matlab中插值函数汇总和使用说明
http://blog.sciencenet.cn/blog-457143-679275.html MATLAB中的插值函数为interp1,其调用格式为: yi= interp1(x,y,xi,' ...
- elif 相当于else&if
if 条件: 语句块 elif 条件: 语句块 ... else #elif好像要有一个else作为结尾
- Substring Search
查找子字符串 Introduction 在长度为 N 的文本里寻找长度为 M 的模式(子串),典型情况是 N >> M. 这个应用就很广泛啦,在文本中寻找特定的模式(子串)是很常见的需求. ...
- python第十四课--排序及自定义函数之案例二:冒泡排序
案例二:冒泡排序 lt1=[45,12,56,-32,-3,44,75,-22,100] print('排序前:'+str(lt1)) 自定义函数:实现冒泡排序(升序)原则:1).有没有形参?有,接受 ...
- HtmlImageGenerator字体乱码问题解决、html2image放linux上乱码问题解决
使用html2image-0.9.jar生成图片. 在本地window系统正常,放到服务器linux系统时候中文乱码问题.英文可以,中文乱码应该就是字体问题了. 一.首先需要在linux安装字体,si ...
- Day6 jQuery
元素的操作 dom对象和jQuery对象 dom对象:原生js获取节点 jQuery对象:通过jQuery获取节点对象 //dom对象 var oP = document.getElementById ...