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中的所有元素。

Output

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

Sample Input

4 3 1 2
1 2

Sample Output

8

HINT

【样例说明】
可以生成的满足要求的不同的数列有(1,1,1,1)、(1,1,2,2)、(1,2,1,2)、(1,2,2,1)、(2,1,1,2)、(2,1,2,1)、(2,2,1,1)、(2,2,2,2)。
【数据规模和约定】
对于10%的数据,1<=N<=1000;
对于30%的数据,3<=M<=100;
对于60%的数据,3<=M<=800;
对于全部的数据,1<=N<=10^9,3<=M<=8000,M为质数,1<=x<=M-1,输入数据保证集合S中元素不重复
 
好厉害。。。
首先有10分算法,设f[i][j]表示选i个数,乘积模M结果为x的方案数。
然后因为M为质数,我们可以求出M的原根g,这样转移就可以写成关于g的生成函数,就可以用NTT来加速了。
接下来发现每次乘的都是相同的多项式,那么多项式快速幂即可。
  1. #include<cstdio>
  2. #include<cctype>
  3. #include<queue>
  4. #include<cstring>
  5. #include<algorithm>
  6. #define rep(i,s,t) for(int i=s;i<=t;i++)
  7. #define dwn(i,s,t) for(int i=s;i>=t;i--)
  8. #define ren for(int i=first[x];i;i=next[i])
  9. using namespace std;
  10. const int BufferSize=1<<16;
  11. char buffer[BufferSize],*head,*tail;
  12. inline char Getchar() {
  13. if(head==tail) {
  14. int l=fread(buffer,1,BufferSize,stdin);
  15. tail=(head=buffer)+l;
  16. }
  17. return *head++;
  18. }
  19. inline int read() {
  20. int x=0,f=1;char c=Getchar();
  21. for(;!isdigit(c);c=Getchar()) if(c=='-') f=-1;
  22. for(;isdigit(c);c=Getchar()) x=x*10+c-'0';
  23. return x*f;
  24. }
  25. const int maxn=18010;
  26. const int p=1004535809;
  27. const int G=3;
  28. const int NUM=15;
  29. typedef long long ll;
  30. ll wn[maxn],inv;
  31. ll pow(ll n,ll m,ll mod=p) {
  32. ll ans=1;
  33. for(;m;m>>=1,(n*=n)%=mod) if(m&1) (ans*=n)%=mod;
  34. return ans;
  35. }
  36. void NTT(ll* A,int len,int tp) {
  37. int j=len>>1,c=0;
  38. rep(i,1,len-2) {
  39. if(i<j) swap(A[i],A[j]);int k=len>>1;
  40. while(j>=k) j-=k,k>>=1;j+=k;
  41. }
  42. for(int i=2;i<=len;i<<=1) {
  43. c++;
  44. for(int j=0;j<len;j+=i) {
  45. ll w=1;
  46. for(int k=j;k<j+(i>>1);k++) {
  47. ll u=A[k],t=w*A[k+(i>>1)]%p;
  48. A[k]=(u+t)%p;A[k+(i>>1)]=(u-t+p)%p;
  49. (w*=wn[c])%=p;
  50. }
  51. }
  52. }
  53. if(tp<0) {
  54. rep(i,1,len/2-1) swap(A[i],A[len-i]);
  55. ll inv=pow(len,p-2);
  56. rep(i,0,len-1) (A[i]*=inv)%=p;
  57. }
  58. }
  59. int check(int g,int m) {
  60. for(int i=2;i*i<m;i++) if((m-1)%i==0&&(pow(g,i,m)==1||pow(g,(m-1)/i,m)==1)) return 0;
  61. return 1;
  62. }
  63. int n,m,X,S,len,gs=2,c[maxn];
  64. ll D[maxn];
  65. void mul(ll* A,ll* B) {
  66. rep(i,0,len-1) D[i]=B[i];
  67. NTT(A,len,1);NTT(D,len,1);
  68. rep(i,0,len-1) (A[i]*=D[i])%=p;
  69. NTT(A,len,-1);
  70. dwn(i,len-1,m-1) (A[i-m+1]+=A[i])%=p,A[i]=0;
  71. }
  72. ll A[maxn],B[maxn];
  73. int main() {
  74. rep(i,0,NUM-1) wn[i]=pow(G,(p-1)/(1<<i));
  75. n=read();m=read();X=read();S=read();
  76. len=1;while(len<=(m<<1)) len<<=1;
  77. while(!check(gs,m)) gs++;
  78. int m0=1;c[1]=0;
  79. rep(i,1,m-2) (m0*=gs)%=m,c[m0]=i;
  80. rep(i,1,S) {
  81. int x=read()%m;
  82. if(x) A[c[x]]=1;
  83. }
  84. B[0]=1;
  85. for(;n;mul(A,A),n>>=1) if(n&1) mul(B,A);
  86. printf("%lld\n",B[c[X]]);
  87. return 0;
  88. }

  

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]序列统计 10分

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

  5. 【NTT】bzoj3992: [SDOI2015]序列统计

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

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

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

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

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

  8. BZOJ 3992: [SDOI2015]序列统计 [快速数论变换 生成函数 离散对数]

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

  9. [SDOI2015]序列统计

    [SDOI2015]序列统计 标签: NTT 快速幂 Description 给你一个模m意义下的数集,需要用这个数集生成一个数列,使得这个数列在的乘积为x. 问方案数模\(1004535809\). ...

随机推荐

  1. 查看Linux分区格式

    第一种方法: 使用mount   [root@ol6-121-rac1 ~]# mount /dev/mapper/vg_ol6121rac1-lv_root on / type ext4 (rw) ...

  2. Install PIL on mac osX10.9

    follow this instruction: http://blog.csdn.net/liushuaikobe/article/details/8729652 and if you encoun ...

  3. 攻城狮在路上(叁)Linux(二十九)--- 完整备份工具:dump以及restore

    一.dump命令: 该命令既可以针对整个文件系统进行备份,也可以仅针对目录来备份.还可以指定不同的备份等级(-0~-9共10个等级). dump -W:列出在/etc/fstab中具有dump设置的分 ...

  4. AOP常用术语

    1.连接点(Joinpoint) 程序执行的某个特定位置:如类开始初始化前,类初始化后,类某个方法调用前,调用后,方法跑出异常后.一个类或一段程序代码拥有一些具有边界性质的特定点.这些代码中的特定点就 ...

  5. iphone手机不同版本兼容、横竖屏

    /* 兼容问题*/ @media screen and (device-width: 320px) and (device-height: 480px) and (-webkit-device-pix ...

  6. JQuery 操作对象的属性值

    通过JQuery去操作前台对象(div,span...)的属性是很常见的事情,本文就简单的介绍几种操作情形. 1):通过属性值去获取对象 2):用JQuery去修改对象的属性值 3):获取并修改对象的 ...

  7. ASP.NET MVC使用过滤器进行权限控制

    1.新建MVC项目 2.找到Models文件夹,新建 LoginCheckFilterAttribute 类 public class LoginCheckFilterAttribute : Acti ...

  8. Android自动化压力测试之Monkey Test Android常见的错误类型及黑白名单的使用方法(四)

    Android常见的错误类型有两种 1.ANR类型 1)在5秒内没有响应输入的事件(例如,按键按下,屏幕触摸) 2)BroadcastReceiver在10秒内没有执行完毕 2.Crash类型 1)异 ...

  9. 为什么调用 FragmentPagerAdapter.notifyDataSetChanged() 并不能更新其 Fragment?

    在一个 Android 应用中,我使用 FragmentPagerAdapter 来 处理多 Fragment 页面的横向滑动.不过我碰到了一个问题,即当 Fragment 对应的数据集发生改变时,我 ...

  10. linux服务器init 5启动图形界面,报错Retrigger failed udev events

    今天因工作需要开启linux系统的桌面环境,使用startx未成功,报如下错误: [root@ /]# startx xauth: creating new authority xinit: No s ...