板子题都差点不会了

Description

小C有一个集合S,里面的元素都是小于M的非负整数。他用程序编写了一个数列生成器,可以生成一个长度为N的数
列,数列中的每个数都属于集合S。小C用这个生成器生成了许多这样的数列。但是小C有一个问题需要你的帮助:
给定整数x,求所有可以生成出的,且满足数列中所有数的乘积mod M的值等于x的不同的数列的有多少个。小C认为
,两个数列{Ai}和{Bi}不同,当且仅当至少存在一个整数i,满足Ai≠Bi。另外,小C认为这个问题的答案可能很大
,因此他只需要你帮助他求出答案mod 1004535809的值就可以了。

Input

一行,四个整数,N、M、x、|S|,其中|S|为集合S中元素个数。
第二行,|S|个整数,表示集合S中的所有元素。
1<=N<=10^9,3<=M<=8000,M为质数
0<=x<=M-1,输入数据保证集合S中元素不重复x∈[1,m-1]

集合中的数∈[0,m-1]

Output

一行,一个整数,表示你求出的种类数mod 1004535809的值。


题目分析

用$f_{i,j}$表示选了$i$个数乘积为$j$的方案数,不难得到式子$f_{2i,j}=\sum\limits_{a*b\equiv j(\mod m)}f_{i,a}\times f_{i,b}$

对于乘法转加法,常见套路就是取对数。那么这里在模意义下,注意到原根$g$有很好的性质:$g^0,g^1,\cdots ,g^{m-2}$可以取遍$[1\cdots m-1]$,因此令$j\equiv g^A(\mod m)$,不妨用$A$代替$j$,以此类推。所以剩下的就是一个多项式快速幂的过程了,最终答案就是$f_x$的系数。

中途写错两个地方:为$[1\cdots m-1]$内元素按$g^i$标号时,写成遍历$[0\cdots m-1]$(这里整体下标减一。说到底还是对式子不熟练。);给$x$重标号时候,写成 if (x==tar&&!fl) tar = x, fl = ; ……

 #include<bits/stdc++.h>
#define MO 1004535809
const int maxn = ; int T,n,m,tar,len,dt,inv3,invn,ort;
int f[maxn],g[maxn],cov[maxn],tmp1[maxn],tmp2[maxn];
bool vis[maxn]; int read()
{
char ch = getchar();
int num = , fl = ;
for (; !isdigit(ch); ch=getchar())
if (ch=='-') fl = -;
for (; isdigit(ch); ch=getchar())
num = (num<<)+(num<<)+ch-;
return num*fl;
}
int qmi(int a, int b, int p)
{
int ret = ;
for (; b; b>>=, a=1ll*a*a%p)
if (b&) ret = 1ll*ret*a%p;
return ret;
}
int fndRoot()
{
int fac[], tot = , pos = m-;
for (int i=; i*i<=pos; i++)
if (pos%i==){
fac[++tot] = i;
while (pos%i==) pos /= i;
}
if (pos!=) fac[++tot] = pos;
pos = m-;
for (int i=; i<=pos; i++)
{
bool chk = true;
for (int j=; j<=tot&&chk; j++)
if (qmi(i, pos/fac[j], m)==) chk = false;
if (chk) return i;
}
return -;
}
void NTT(int *a, int opt)
{
for (int i=; i<len; i++)
if (i < cov[i]) std::swap(a[i], a[cov[i]]);
for (int i=; i<len; i<<=)
{
int Wn = qmi(, (MO-)/(i<<), MO);
if (opt==-) Wn = qmi(inv3, (MO-)/(i<<), MO);
for (int j=, p=i<<; j<len; j+=p)
{
int w = ;
for (int k=; k<i; k++, w=1ll*w*Wn%MO)
{
int valx = a[j+k], valy = 1ll*w*a[i+j+k]%MO;
a[j+k] = (valx+valy)%MO, a[i+j+k] = (valx-valy+MO)%MO;
}
}
}
if (opt==-) for (int i=; i<len; i++) a[i] = 1ll*a[i]*invn%MO;
}
void mult(int *a, int *b)
{
for (int i=; i<len; i++) tmp1[i] = a[i], tmp2[i] = b[i];
NTT(tmp1, ), NTT(tmp2, );
for (int i=; i<len; i++) tmp1[i] = 1ll*tmp1[i]*tmp2[i]%MO;
NTT(tmp1, -);
for (int i=; i<m-; i++) a[i] = (tmp1[i]+tmp1[i+m-])%MO;
}
int main()
{
T = read(), m = read(), tar = read(), n = read();
inv3 = qmi(, MO-, MO), ort = fndRoot();
for (int i=; i<=n; i++) vis[read()] = true;
for (int i=, x=, fl=; i<m-; i++, x=1ll*x*ort%m)
{
if (vis[x]) g[i] = ;
if (x==tar&&!fl) tar = i, fl = ;
}
for (len=; len<=(m-)<<; len<<=) ++dt;
for (int i=; i<len; i++)
cov[i] = (cov[i>>]>>)|((i&)<<(dt-));
f[] = , invn = qmi(len, MO-, MO);
for (; T; T>>=, mult(g, g))
if (T&) mult(f, g);
printf("%d\n",f[tar]);
return ;
}

END

【NTT】bzoj3992: [SDOI2015]序列统计的更多相关文章

  1. [BZOJ3992][SDOI2015]序列统计(DP+原根+NTT)

    3992: [SDOI2015]序列统计 Time Limit: 30 Sec  Memory Limit: 128 MBSubmit: 1888  Solved: 898[Submit][Statu ...

  2. BZOJ3992: [SDOI2015]序列统计(NTT 原根 生成函数)

    题意 题目链接 给出大小为\(S\)的集合,从中选出\(N\)个数,满足他们的乘积\(\% M = X\)的方案数 Sol 神仙题Orz 首先不难列出最裸的dp方程,设\(f[i][j]\)表示选了\ ...

  3. 2018.12.31 bzoj3992: [SDOI2015]序列统计(生成函数+ntt+快速幂)

    传送门 生成函数简单题. 题意:给出一个集合A={a1,a2,...as}A=\{a_1,a_2,...a_s\}A={a1​,a2​,...as​},所有数都在[0,m−1][0,m-1][0,m− ...

  4. BZOJ3992: [SDOI2015]序列统计

    Description 小C有一个集合S,里面的元素都是小于M的非负整数.他用程序编写了一个数列生成器,可以生成一个长度为N的数列,数列中的每个数都属于集合S. 小C用这个生成器生成了许多这样的数列. ...

  5. BZOJ3992 [SDOI2015]序列统计 【生成函数 + 多项式快速幂】

    题目 小C有一个集合S,里面的元素都是小于M的非负整数.他用程序编写了一个数列生成器,可以生成一个长度为N的数 列,数列中的每个数都属于集合S.小C用这个生成器生成了许多这样的数列.但是小C有一个问题 ...

  6. 【动态规划】bzoj3992 [Sdoi2015]序列统计 10分

    #include<cstdio> using namespace std; #define MOD 1004535809 int a[8001],f[1001][101],n,m,x,S; ...

  7. 【BZOJ3992】[SDOI2015]序列统计 NTT+多项式快速幂

    [BZOJ3992][SDOI2015]序列统计 Description 小C有一个集合S,里面的元素都是小于M的非负整数.他用程序编写了一个数列生成器,可以生成一个长度为N的数列,数列中的每个数都属 ...

  8. 【BZOJ3992】序列统计(动态规划,NTT)

    [BZOJ3992]序列统计(动态规划,NTT) 题面 BZOJ 题解 最裸的暴力 设\(f[i][j]\)表示前\(i\)个数,积在膜意义下是\(j\)的方案数 转移的话,每次枚举一个数,直接丢进去 ...

  9. BZOJ 3992: [SDOI2015]序列统计 NTT+快速幂

    3992: [SDOI2015]序列统计 Time Limit: 30 Sec  Memory Limit: 128 MBSubmit: 1155  Solved: 532[Submit][Statu ...

随机推荐

  1. sourcetree基本使用

    非常有用的使用sourcetree开发的步骤文档 https://www.cnblogs.com/fps2tao/p/7825742.html 1) master,最终发布版本,整个项目中有且只有一个 ...

  2. codeforces C. Vasya And The Mushrooms (思维+模拟)

    题意:给定一个2*n的矩形方格,每个格子有一个权值,从(0,0)开始出发,要求遍历完整个网格(不能重复走一个格子),求最大权值和,(权值和是按照step*w累加,step步数从0开始). 转载: 题解 ...

  3. Django - CRM项目(2)Q查询(模糊查询)

    一.CRM项目(2) 利用Q查询中的q对象完成条件筛选功能. 批量删除.公户转私户功能. 新增一张跟进记录表ConsultRecord,迁移数据库并添加测试数据,实现跟进记录列表页面. 客户列表新增跟 ...

  4. mysql /etc/my.cnf

    [client] port=3306 socket = /tmp/mysql.sock default-character-set=utf8 [mysqld] # 1 general config p ...

  5. jquery——制作置顶菜单

    置顶菜单: <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF ...

  6. LeetCode 232 Implement Queue using Stacks 两个栈实现队列

    class MyQueue { public: /** Initialize your data structure here. */ MyQueue() { } /** Push element x ...

  7. win10的一些设置

    win10进入安全模式: 左下角->设置->更新和安全->恢复->立即重启  (重启后进入的界面可以进行一些设置来进入安全模式) win10切换账户: 任务管理器->账户

  8. javascript模块化编程规范

    一.javascript模块化编程规范: 二.关于commenjs规范和AMD规范: 根本不同:前者用于服务器端同步加载模块:后者是客户端异步加载模块. 同点:两者都有一个全局函数require(), ...

  9. C++中的swap函数

    最通用的模板交换函数模式:创建临时对象,调用对象的赋值操作符 template <class T> void swap ( T& a, T& b ) { T c(a); a ...

  10. swift第一课快速体验playground

    最近听说苹果要大力推行swift语言,所以我必须要赶快好好学一学,今天做第一个就遇到问题. 在Xcode7.2欢迎界面,选中创建第一个,我们一般都是默认创建第二个. 创建完后,出现问题了,提示如下: ...