题面

传送门

题解

首先你得会多项式开根->这里

其次你得会解形如

\[x^2\equiv a \pmod{p}
\]

的方程

这里有两种方法,一个是\(bsgs\)(这里),还有一种是\(Cipolla\)(这里)(不过这个只能用来解二次剩余就是了)

代码里留着的是\(bsgs\),注释掉的是\(Cipolla\)

如果用\(Cipolla\)的话注意这里需要求的是较小的那个解

  1. //minamoto
  2. #include<bits/stdc++.h>
  3. #include<tr1/unordered_map>
  4. #define R register
  5. #define fp(i,a,b) for(R int i=(a),I=(b)+1;i<I;++i)
  6. #define fd(i,a,b) for(R int i=(a),I=(b)-1;i>I;--i)
  7. #define go(u) for(int i=head[u],v=e[i].v;i;i=e[i].nx,v=e[i].v)
  8. using namespace std;
  9. using namespace std::tr1;
  10. char buf[1<<21],*p1=buf,*p2=buf;
  11. inline char getc(){return p1==p2&&(p2=(p1=buf)+fread(buf,1,1<<21,stdin),p1==p2)?EOF:*p1++;}
  12. int read(){
  13. R int res,f=1;R char ch;
  14. while((ch=getc())>'9'||ch<'0')(ch=='-')&&(f=-1);
  15. for(res=ch-'0';(ch=getc())>='0'&&ch<='9';res=res*10+ch-'0');
  16. return res*f;
  17. }
  18. char sr[1<<21],z[20];int C=-1,Z=0;
  19. inline void Ot(){fwrite(sr,1,C+1,stdout),C=-1;}
  20. void print(R int x){
  21. if(C>1<<20)Ot();if(x<0)sr[++C]='-',x=-x;
  22. while(z[++Z]=x%10+48,x/=10);
  23. while(sr[++C]=z[Z],--Z);sr[++C]=' ';
  24. }
  25. const int N=(1<<18)+5,P=998244353,inv2=499122177;
  26. inline int add(R int x,R int y){return x+y>=P?x+y-P:x+y;}
  27. inline int dec(R int x,R int y){return x-y<0?x-y+P:x-y;}
  28. inline int mul(R int x,R int y){return 1ll*x*y-1ll*x*y/P*P;}
  29. int ksm(R int x,R int y){
  30. R int res=1;
  31. for(;y;y>>=1,x=mul(x,x))(y&1)?res=mul(res,x):0;
  32. return res;
  33. }
  34. unordered_map<int,int>mp;
  35. int bsgs(int x){
  36. int m=sqrt(P)+1;mp.clear();
  37. for(R int i=0,res=x;i<m;++i,res=mul(res,3))mp[res]=i;
  38. for(R int i=1,tmp=ksm(3,m),res=tmp;i<=m;++i,res=mul(res,tmp))
  39. if(mp.count(res))return i*m-mp[res];
  40. }
  41. int exgcd(int a,int b,int &x,int &y){
  42. if(!b)return x=1,y=0,a;
  43. int d=exgcd(b,a%b,y,x);
  44. return y-=a/b*x,d;
  45. }
  46. int Get(int p){
  47. int c=bsgs(p),x,y,d=exgcd(2,P-1,x,y);
  48. if(c%d)return -1;
  49. int t=abs((P-1)/d);x=(1ll*x*c/d%t+t)%t;
  50. return ksm(3,x);
  51. }
  52. //struct cp{
  53. // int x,y;
  54. // inline cp(R int xx,R int yy):x(xx),y(yy){}
  55. //};
  56. //inline cp mul(R cp a,R cp b,R int w){return cp(add(mul(a.x,b.x),mul(w,mul(a.y,b.y))),add(mul(a.x,b.y),mul(a.y,b.x)));}
  57. //int ksm(R cp x,R int y,R int w){
  58. // cp res(1,0);
  59. // for(;y;y>>=1,x=mul(x,x,w))if(y&1)res=mul(res,x,w);
  60. // return res.x;
  61. //}
  62. //int Get(int p){
  63. // srand(time(0));
  64. // if(ksm(p,(P-1)>>1)==P-1)return -1;
  65. // while(true){
  66. // int a=mul(rand(),rand()),w=dec(mul(a,a),p);
  67. // if(ksm(w,(P-1)>>1)==P-1){
  68. // int x=ksm(cp(a,1),(P+1)>>1,w);
  69. // return min(x,P-x);
  70. // }
  71. // }
  72. //}
  73. int r[21][N],w[2][N],lg[N],inv[21];
  74. void Pre(){
  75. fp(d,1,18){
  76. fp(i,0,(1<<d)-1)r[d][i]=(r[d][i>>1]>>1)|((i&1)<<(d-1));
  77. lg[1<<d]=d,inv[d]=ksm(1<<d,P-2);
  78. }
  79. for(R int t=(P-1)>>1,i=1,x,y;i<262144;i<<=1,t>>=1){
  80. x=ksm(3,t),y=ksm(332748118,t),w[0][i]=w[1][i]=1;
  81. fp(k,1,i-1)
  82. w[1][i+k]=mul(w[1][i+k-1],x),
  83. w[0][i+k]=mul(w[0][i+k-1],y);
  84. }
  85. }
  86. int lim,d;
  87. void NTT(int *A,int ty){
  88. fp(i,0,lim-1)if(i<r[d][i])swap(A[i],A[r[d][i]]);
  89. for(R int mid=1;mid<lim;mid<<=1)
  90. for(R int j=0,t;j<lim;j+=(mid<<1))
  91. fp(k,0,mid-1)
  92. A[j+k+mid]=dec(A[j+k],t=mul(w[ty][mid+k],A[j+k+mid])),
  93. A[j+k]=add(A[j+k],t);
  94. if(!ty)fp(i,0,lim-1)A[i]=mul(A[i],inv[d]);
  95. }
  96. void Inv(int *a,int *b,int len){
  97. if(len==1)return b[0]=ksm(a[0],P-2),void();
  98. Inv(a,b,len>>1);
  99. static int A[N],B[N];lim=(len<<1),d=lg[lim];
  100. fp(i,0,len-1)A[i]=a[i],B[i]=b[i];
  101. fp(i,len,lim-1)A[i]=B[i]=0;
  102. NTT(A,1),NTT(B,1);
  103. fp(i,0,lim-1)A[i]=mul(A[i],mul(B[i],B[i]));
  104. NTT(A,0);
  105. fp(i,0,len-1)b[i]=dec(add(b[i],b[i]),A[i]);
  106. fp(i,len,lim-1)b[i]=0;
  107. }
  108. void Sqrt(int *a,int *b,int len){
  109. if(len==1)return b[0]=Get(a[0]),void();
  110. Sqrt(a,b,len>>1);
  111. static int A[N],B[N];
  112. fp(i,0,len-1)A[i]=a[i];Inv(b,B,len);
  113. lim=(len<<1),d=lg[lim];fp(i,len,lim-1)A[i]=B[i]=0;
  114. NTT(A,1),NTT(B,1);
  115. fp(i,0,lim-1)A[i]=mul(A[i],B[i]);
  116. NTT(A,0);
  117. fp(i,0,len-1)b[i]=mul(add(b[i],A[i]),inv2);
  118. fp(i,len,lim-1)b[i]=0;
  119. }
  120. int A[N],B[N],n;
  121. int main(){
  122. // freopen("testdata.in","r",stdin);
  123. n=read(),Pre();
  124. fp(i,0,n-1)A[i]=read();
  125. int len=1;while(len<n)len<<=1;
  126. Sqrt(A,B,len);
  127. fp(i,0,n-1)print(B[i]);
  128. return Ot(),0;
  129. }

P5277 【模板】多项式开根(加强版)(bsgs or Cipolla)的更多相关文章

  1. FFT模板 生成函数 原根 多项式求逆 多项式开根

    FFT #include<iostream> #include<cstring> #include<cstdlib> #include<cstdio> ...

  2. CF438E The Child and Binary Tree(生成函数+多项式开根+多项式求逆)

    传送门 可以……这很多项式开根模板……而且也完全不知道大佬们怎么把这题的式子推出来的…… 首先,这题需要多项式开根和多项式求逆.多项式求逆看这里->这里,这里讲一讲多项式开根 多项式开方:已知多 ...

  3. Codeforces 250 E. The Child and Binary Tree [多项式开根 生成函数]

    CF Round250 E. The Child and Binary Tree 题意:n种权值集合C, 求点权值和为1...m的二叉树的个数, 形态不同的二叉树不同. 也就是说:不带标号,孩子有序 ...

  4. 【XSY2730】Ball 多项式exp 多项式ln 多项式开根 常系数线性递推 DP

    题目大意 一行有\(n\)个球,现在将这些球分成\(k\) 组,每组可以有一个球或相邻两个球.一个球只能在至多一个组中(可以不在任何组中).求对于\(1\leq k\leq m\)的所有\(k\)分别 ...

  5. 【BZOJ3625】【CF438E】小朋友和二叉树 NTT 生成函数 多项式开根 多项式求逆

    题目大意 考虑一个含有\(n\)个互异正整数的序列\(c_1,c_2,\ldots ,c_n\).如果一棵带点权的有根二叉树满足其所有顶点的权值都在集合\(\{c_1,c_2,\ldots ,c_n\ ...

  6. BZOJ3625 [Codeforces Round #250]小朋友和二叉树(生成函数+多项式开根)

    设f(n)为权值为n的神犇二叉树个数.考虑如何递推求这个东西. 套路地枚举根节点的左右子树.则f(n)=Σf(i)f(n-i-cj),cj即根的权值.卷积的形式,cj也可以通过卷上一个多项式枚举.可以 ...

  7. 【BZOJ3625】【codeforces438E】小朋友和二叉树 生成函数+多项式求逆+多项式开根

    首先,我们构造一个函数$G(x)$,若存在$k∈C$,则$[x^k]G(x)=1$. 不妨设$F(x)$为最终答案的生成函数,则$[x^n]F(x)$即为权值为$n$的神犇二叉树个数. 不难推导出,$ ...

  8. BZOJ 3625 [Codeforces Round #250]小朋友和二叉树 ——NTT 多项式求逆 多项式开根

    生成函数又有奇妙的性质. $F(x)=C(x)*F(x)*F(x)+1$ 然后大力解方程,得到一个带根号的式子. 多项式开根有解只与常数项有关. 发现两个解只有一个是成立的. 然后多项式开根.求逆. ...

  9. [BZOJ 3625] [Codeforces 438E] 小朋友的二叉树 (DP+生成函数+多项式开根+多项式求逆)

    [BZOJ 3625] [Codeforces 438E] 小朋友的二叉树 (DP+生成函数+多项式开根+多项式求逆) 题面 一棵二叉树的所有点的点权都是给定的集合中的一个数. 让你求出1到m中所有权 ...

随机推荐

  1. ArrayList原理(一)

    需要使用到动态数组的时候用的最多的就是ArrayList了,底层其实是Object数组,以下demo基于JDK1.8: List<Integer> list  = new ArrayLis ...

  2. C#格式化数字

    var t1 = Profiler.GetMonoHeapSize()/div; var t2 = Profiler.GetMonoUsedSize() / div; var t3 = Profile ...

  3. hive 建表导入数据

    1. hive> create table wyp > (id int, name string, > age int, tel string) > ROW FORMAT DE ...

  4. 服务级别协议(SLA)与运行水平协议(OLA)

    服务级别协议(SLA)与运行水平协议(OLA): 服务级别管理和服务级别协议在国内已被广泛接受并成用.本文试图讨论服务级别协议(SLA)和运行水平协议(OLA)的异同. 1. SLA a.定义 服务级 ...

  5. C#获取MAC地址的几种方法

    首先需要用到的一些方法和类: public enum NCBCONST { NCBNAMSZ = 16, MAX_LANA = 254, NCBENUM = 0x37, NRC_GOODRET = 0 ...

  6. python全栈考试

    1.执行 Python 脚本的两种方式 shell直接调用python脚本 python run.py 调用解释器来调用脚本  2.2.简述位.字节的关系 每8个位bit,组成一个字节byte. 一个 ...

  7. 启动memcached

    /usr/local/bin/memcached -d -c -m -u root

  8. SQLServer函数 left()、charindex()、stuff()的使用

    1.left() LEFT (<character_expression>, <integer_expression>)   返回character_expression 左起 ...

  9. 借用服务器百度BAE

    3一.简介 对于普通的开发者,不必要买服务器和买域名,这时要将自己的项目传到服务器上,就用到了百度BAE这样的,可以直接传项目的服务器. 二.申请 登录百度开放平台上 三.登录网址,选择要使用的项目 ...

  10. CPU位数、地址线位数、数据线位数、通用寄存器位数!

    CPU位数:表示的是其通用寄存器的位数,CPU的位数表示该CPU一次处理数据的最大位数. 数据线位数:是CPU的理论最大寻址空间,也是CPU与内存之间一次最大的数据传输位数. 地址线位数:是CPU实际 ...