[SDOI 2015]序列统计
Description
给出集合 \(S\) ,元素都是小于 \(M\) 的非负整数。问能够生成出多少个长度为 \(N\) 的数列 \(A\) ,数列中的每个数都属于集合 \(S\) ,并且
\[\prod_{i=1}^N A_i\equiv x \pmod{M}\]
答案对 \(1004535809\) 取模。
\(1\leq N\leq 10^9,3\leq M\leq 8000, M 为质数,0\leq x\leq M-1\)
Solution
显然能够得到 \(DP\) 的解法:令 \(f_{i,j}\) 为生成序列长度为 \(i\) 时,乘积在模 \(M\) 意义下为 \(j\) 的方案数。
显然 \(f_{i,j}\rightarrow f_{i+1,(j\times w)\mod M},w\in S\) 。
但 \(n\leq 10^9\) 显然不能递推。考虑优化。
由于乘法不太好搞,我们试着换种思路,我们不妨将集合内数取 \(\log\) 。那么 \(f_{i,\log j}\rightarrow f_{i+1,\log j+\log w},w\in S\) 。
但实数域上确实不好做,考虑取离散对数。由费马小定理,它是以 \(M-1\) 为周期的,那么只要 \(\text{NTT}\) 优化,加上快速幂。对模意义外的数讨论即可。
Code
#include <bits/stdc++.h>
using namespace std;
const int yzh = 1004535809;
const int N = 8000*4;
int n, m, x, s, G, lg[N+5], a, len, L, R[N+5];
int A[N+5];
int quick_pow(int a, int b, int yzh) {
int ans = 1;
while (b) {
if (b&1) ans = 1ll*a*ans%yzh;
b >>= 1, a = 1ll*a*a%yzh;
}
return ans;
}
void get_G() {
int prime[N+5], tot = 0, x = m-1;
for (int i = 2, lim = sqrt(x)+1; i <= lim; i++)
if (x%i == 0) {
prime[++tot] = i;
while (x%i == 0) x /= i;
}
if (x != 1) prime[++tot] = x;
for (int i = 2; true; i++) {
int flag = 1;
for (int j = 1; j <= tot; j++)
if (quick_pow(i, (m-1)/prime[j], m) == 1) {
flag = 0; break;
}
if (flag == 1) {G = i; break; }
}
for (int i = 1, g = G; i < m; i++, g = 1ll*g*G%m) lg[g] = i;
}
void NTT(int *A, int o) {
for (int i = 0; i < len; i++) if (i < R[i]) swap(A[i], A[R[i]]);
for (int i = 1; i < len; i <<= 1) {
int gn = quick_pow(3, (yzh-1)/(i<<1), yzh), x, y;
if (o == -1) gn = quick_pow(gn, yzh-2, yzh);
for (int j = 0; j < len; j += (i<<1)) {
int g = 1;
for (int k = 0; k < i; k++, g = 1ll*g*gn%yzh) {
x = A[j+k], y = 1ll*g*A[j+k+i]%yzh;
A[j+k] = (x+y)%yzh, A[j+k+i] = (x-y+yzh)%yzh;
}
}
}
if (o == 1) return;
for (int i = 0, inv = quick_pow(len, yzh-2, yzh); i < len; i++)
A[i] = 1ll*A[i]*inv%yzh;
for (int i = m; i < len; i++) (A[i%(m-1) ? i%(m-1) : m-1] += A[i]) %= yzh, A[i] = 0;
}
void NTTpow(int *A, int b) {
int ans[N+5] = {0}; ans[0] = 1;
while (b) {
NTT(A, 1);
if (b&1) {
NTT(ans, 1);
for (int i = 0; i < len; i++) ans[i] = 1ll*ans[i]*A[i]%yzh;
NTT(ans, -1);
}
for (int i = 0; i < len; i++) A[i] = 1ll*A[i]*A[i]%yzh;
NTT(A, -1); b >>= 1;
}
for (int i = 0; i < len; i++) A[i] = ans[i];
}
void work() {
scanf("%d%d%d%d", &n, &m, &x, &s); get_G();
for (int i = 1; i <= s; i++) {scanf("%d", &a); ++A[lg[a]]; }
A[0] = 0;
for (len = 1; len <= (m<<1); len <<= 1) ++L;
for (int i = 0; i < len; i++) R[i] = (R[i>>1]>>1)|((i&1)<<(L-1));
NTTpow(A, n); printf("%d\n", A[lg[x]]);
}
int main() {work(); return 0; }
[SDOI 2015]序列统计的更多相关文章
- [BZOJ 3992] [SDOI 2015] 序列统计(DP+原根+NTT)
[BZOJ 3992] [SDOI 2015] 序列统计(DP+原根+NTT) 题面 小C有一个集合S,里面的元素都是小于质数M的非负整数.他用程序编写了一个数列生成器,可以生成一个长度为N的数列,数 ...
- BZOJ 3992 [SDOI 2015] 序列统计 解题报告
这个题最暴力的搞法就是这样的: 设 $Dp[i][j]$ 为前 $i$ 个数乘积为 $j$ 的方案数. 转移的话就不多说了哈... 当前复杂度 $O(nm^2)$ 注意到,$M$ 是个质数,就说明 $ ...
- [BZOJ 3992] [SDOI 2015] 序列统计
Description 传送门 Solution [一] 设 \(f[i][j]\) 表示前 \(i\) 个数的乘积在模 \(p\) 意义下等于 \(j\) 的方案数,有 \[ f[i][j]=\su ...
- [BZOJ 3992][SDOI2015]序列统计
3992: [SDOI2015]序列统计 Time Limit: 30 Sec Memory Limit: 128 MBSubmit: 2275 Solved: 1090[Submit][Stat ...
- Mobius反演与积性函数前缀和演学习笔记 BZOJ 4176 Lucas的数论 SDOI 2015 约数个数和
下文中所有讨论都在数论函数范围内开展. 数论函数指的是定义域为正整数域, 且值域为复数域的函数. 数论意义下的和式处理技巧 因子 \[ \sum_{d | n} a_d = \sum_{d | n} ...
- Bzoj 4403: 序列统计 Lucas定理,组合数学,数论
4403: 序列统计 Time Limit: 3 Sec Memory Limit: 128 MBSubmit: 328 Solved: 162[Submit][Status][Discuss] ...
- BZOJ4403 序列统计—Lucas你好
绝对是全网写的最详细的一篇题解 题目:序列统计 代码难度:简单 思维难度:提高+-省选 讲下题面:给定三个正整数N.L和R,统计长度在1到N之间,元素大小都在L到R之间的单调不降序列的数量.输出答案 ...
- BZOJ 3992: [SDOI2015]序列统计 [快速数论变换 生成函数 离散对数]
3992: [SDOI2015]序列统计 Time Limit: 30 Sec Memory Limit: 128 MBSubmit: 1017 Solved: 466[Submit][Statu ...
- [SDOI2015]序列统计
[SDOI2015]序列统计 标签: NTT 快速幂 Description 给你一个模m意义下的数集,需要用这个数集生成一个数列,使得这个数列在的乘积为x. 问方案数模\(1004535809\). ...
随机推荐
- 用js来实现那些数据结构(数组篇01)
在开始正式的内容之前,不得不说说js中的数据类型和数据结构,以及一些比较容易让人混淆的概念.那么为什么要从数组说起?数组在js中是最常见的内存数据结构,数组数据结构在js中拥有很多的方法,很多初学者记 ...
- 20155214&20155216 实验二:固件程序设计
---恢复内容开始--- 20155214&20155216 实验二:固件程序设计 实验内容及要求 实验二 固件程序设计-1-MDK 实验要求: 1.注意不经老师允许不准烧写自己修改的代码 2 ...
- 201621123031 《Java程序设计》第12周学习总结
作业12-流与文件 1.本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结多流与文件相关内容. 在Java中的java.io包中定义了许多类专门负责处理各种方式的输入与输出.其中,所有输入 ...
- 【Fungus入门】10分钟快速构建Unity中的万能对话系统 / 叙事系统 / 剧情系统
我真的很久没有写过一个完整的攻略了(笑),咸鱼了很久之后还是想来写一个好玩的.这次主要是梳理一下Unity的小众插件Fungus的核心功能,并且快速掌握其使用方法. 官方文档:http://fungu ...
- 流程控制语句(MySQL/MariaDB )
本文目录:1.BEGIN...END2.true和false3.if结构4.case结构5.loop.leave和iterate6.repeat循环7.while循环 MySQL/MariaDB中的符 ...
- RocketMQ(二):RPC通讯
匠心零度 转载请注明原创出处,谢谢! RocketMQ网络部署图 NameServer:在系统中是做命名服务,更新和发现 broker服务. Broker-Master:broker 消息主机服务器. ...
- Python扩展模块——自动化(testlinkAPI的使用)
使用TESTLINKAPI首先要安装TestLink_API_Python_client-0.6.4(当前最新版本) 目前只使用到了通过api获取testlink中的自定义字段and值 url = ' ...
- Mego(1) - NET中主流ORM框架性能对比
从刚刚开始接触ORM到现在已有超过八年时间,用过了不少ORM框架也了解了不少ORM框架,看过N种关于ORM框架的相关资料与评论,各种言论让人很难选择.在ORM的众多问题中最突出的问题是关于性能方面的问 ...
- CRC 校验
匠心零度 转载请注明原创出处,谢谢! 说明 上篇RocketMQ(二):RPC通讯介绍了rocketmq的一些rpc细节,其实这些内容不仅仅是rocketmq内容,任何通信模块基本都是类似的,这块内容 ...
- redis命令详解
redis中添加key value元素:set key value; 获取元素:get key ; redis中添加集合:lpush key value1 value2 value ...