3283: 运算器

Time Limit: 20 Sec  Memory Limit: 256 MB
Submit: 184  Solved: 59
[Submit][Status][Discuss]

Description

操作有3种:
 

Input

第一行一个正整数N,描述数据组数。
接下来的N行,每行4个正整数Sum,y,z,p。
Sum表述询问类型,如上题所述。对于第2种要求,若X不存在,则输出“Math Error”
 

Output

 
要求有N行输出,每行一个整数,为询问的答案。

Sample Input

4
1 2 10 1000
2 3 1 1000
2 2 3 4
3 2 7 9

Sample Output

24
0
Math Error
3

HINT

操作1个数小于501。保证Y,Z,P小于10^9

操作2个数小于51 保证Y,Z,P小于10^9 P不一定为质数

操作3个数小于51 保证Y,Z小于10^9,P小于10^9

P不一定为质数

P<=10^9

假设分解质因数后,P=p1^s1*p2^s2*……保证pi^ki<=10^5

  VFK上课时提到了这题,然后就试着做了一下。

  本来还以为是水题,结果两个晚上就砸这道题上了,第一问不说了,为啥这类题第一问都是快速幂?第二问本以为可以用经典大步小步做,但是发现由于ax=b(mod c)当解得个数是gcd(a,c),所以gcd(a,c)不等于1是复杂度可以卡到O(c),于是我们可以考虑使gcd(a,c)==1,并且尽量不改变a^x=b (mod c)的解,我们想到了将gcd(a,c)中的因数消去,具体是不断找gcd(a,c),然后借助 a=b (mod c) -> ak=bk (mod ck)的性质,设g=gcd(a,c),将原式变成a*a^(x-1)=b (mod c),发现有解仅当g|a&&g|b&&g|c,同时消去因子,直到互质,这里要特殊处理x较小的解。

  第三问方法是由p是质数的快速阶乘修改而来,这类题通常将p写成PI(pi^ki)的形式,分别计算,在套用中国剩余定理。计算ans=n! mod p [p=pp^pk],将ans表示为k*pp^b的形式。考虑将n!中的数分为与p互质的数,与p不互质的数,对于第一类,由于[1,n),[n+1,2*n)结果相同,可用快速幂优化,对于第二类,同时除以pp可转化为规模n/pp的子问题。

  这道题告诉我以后数论题千万不要用pair存值,稍有修改,一定免不了使用pair<pair<int,int> ,int>这类奇葩类型。

  

  1. #include<iostream>
  2. #include<cstring>
  3. #include<cstdio>
  4. #include<algorithm>
  5. #include<set>
  6. #include<cmath>
  7. #include<vector>
  8. using namespace std;
  9. #define MAXN 1001000
  10. #define INF 0x3f3f3f3f
  11. #define INFL 0x3f3f3f3f3f3f3f3fLL
  12. typedef long long qword;
  13. bool pflag[MAXN];
  14. int prime[MAXN],topp=-;
  15. void init()
  16. {
  17. for (int i=;i<MAXN;i++)
  18. {
  19. if (!pflag[i])
  20. prime[++topp]=i;
  21. for (int j=;j<=topp && i*prime[j]<MAXN;j++)
  22. {
  23. pflag[i*prime[j]]=true;
  24. if (i%prime[j]==)break;
  25. }
  26. }
  27. }
  28. qword fact[MAXN];
  29. qword pow_mod(qword x,qword y,qword mod)
  30. {
  31. qword ret=;
  32. while (y)
  33. {
  34. if (y&)
  35. ret=ret*x%mod;
  36. x=x*x%mod;
  37. y>>=;
  38. }
  39. return ret;
  40. }
  41. qword gcd(qword x,qword y)
  42. {
  43. return x%y==?y:gcd(y,x%y);
  44. }
  45. qword extgcd(qword p,qword q,qword &x,qword &y)
  46. {
  47. if (!q)
  48. {
  49. x=;y=;
  50. return p;
  51. }
  52. qword xx,yy;
  53. qword ret=extgcd(q,p%q,xx,yy);
  54. x=yy;
  55. y=xx-p/q*yy;
  56. return ret;
  57. }
  58.  
  59. pair<int,int> lst[MAXN];
  60. int topl=-;
  61. qword BabyStepGiantStep_Naive(qword k,qword a,qword b,int p)//k*a^x==b (mod p) (gcd(a,p)==1)
  62. {
  63. qword res=INFL;
  64. int sb=(int)ceil(sqrt(p));
  65. qword x=k;
  66. topl=-;
  67. for (int i=;i<sb;i++)
  68. {
  69. lst[++topl].first=(int)x;
  70. lst[topl].second=i;
  71. x=x*a%p;
  72. }
  73. sort(lst,lst+topl+);
  74. qword y=pow_mod(a,sb,p);
  75. x=;
  76. qword xx,yy;
  77. qword r;
  78. for (int i=;i<sb;i++)
  79. {
  80. r=extgcd(x,p,xx,yy);
  81. if (b%r==)
  82. {
  83. xx*=b/r;
  84. xx=(xx%(p/r)+p/r)%(p/r);
  85. if (lower_bound(lst,lst+topl+,make_pair((int)xx,-INF))->first==xx)
  86. {
  87. res=i*sb+lower_bound(lst,lst+topl+,make_pair((int)xx,-INF))->second;
  88. break;
  89. }
  90. }
  91. x=x*y%p;
  92. }
  93. if (res==INFL)
  94. return -;
  95. else
  96. return res;
  97. }
  98. void BabyStepGiantStep(qword a,qword b,int p)//a^x==b (mod P)
  99. {
  100. qword x=%p;
  101. for (int i=;i<;i++)
  102. {
  103. if (x==b)
  104. {
  105. printf("%d\n",i);
  106. return ;
  107. }
  108. x=x*a%p;
  109. }
  110. int delta=;
  111. int g;
  112. qword kk=;
  113. while ((g=(int)gcd(a,p))!=)
  114. {
  115. if (b%g)
  116. {
  117. printf("Math Error\n");
  118. return ;
  119. }
  120. kk*=a/g;
  121. delta++;
  122. b/=g;
  123. p/=g;
  124. kk%=p;
  125. }
  126. qword res=BabyStepGiantStep_Naive(kk,a,b,p);
  127. if (res==-)
  128. printf("Math Error\n");
  129. else
  130. printf("%d\n",(int)(res+delta));
  131. }
  132. void init_fact(int p,int pp)
  133. {
  134. fact[]=;
  135. for (int i=;i<=p;i++)
  136. {
  137. if (i%pp)
  138. {
  139. fact[i]=fact[i-]*i%p;
  140. }
  141. else
  142. {
  143. fact[i]=fact[i-];
  144. int x=i;
  145. while (x%pp==)
  146. {
  147. x/=pp;
  148. }
  149. }
  150. }
  151. }
  152. pair<qword,int> solve2(int n,int p,int pp,int pk)//n! mod p (p=pp^k)
  153. {
  154. pair<qword,int> res;
  155. if (!n)return make_pair(,);
  156. pair<qword,int> pr=solve2(n/pp,p,pp,pk);
  157. res.first=pow_mod(fact[p-],n/p,p) * fact[n%p]%p * pr.first%p;
  158. res.second=n/pp + pr.second;
  159. return res;
  160. }
  161. qword solve4(vector<pair<pair<int,int>,int> > &lst)
  162. {
  163. qword mm=;
  164. for (int i=;i<lst.size();i++)
  165. mm=mm*lst[i].first.first;
  166. qword ans=;
  167. for (int i=;i<lst.size();i++)
  168. {
  169. ans+=mm/lst[i].first.first*pow_mod(mm/lst[i].first.first%lst[i].first.first,lst[i].first.second-,lst[i].first.first)%mm*lst[i].second%mm;
  170. ans%=mm;
  171. }
  172. return ans;
  173. }
  174. int factor[MAXN][];
  175. int totf=;
  176. void solve3(int n,int m,int p)//n!/(m!(n-m)!) mod p
  177. {
  178. if (n<m)
  179. {
  180. printf("0\n");
  181. return ;
  182. }
  183. qword x=p;
  184. totf=;
  185. for (int i=;i<=topp;i++)
  186. {
  187. if (x%prime[i]==)
  188. {
  189. x/=prime[i];
  190. factor[totf][]=prime[i];
  191. factor[totf][]=prime[i];
  192. factor[totf][]=prime[i]-;
  193. factor[totf][]=;
  194. while (x%prime[i]==)
  195. {
  196. factor[totf][]*=prime[i];
  197. factor[totf][]*=prime[i];
  198. factor[totf][]++;
  199. x/=prime[i];
  200. }
  201. totf++;
  202. }
  203. }
  204. if (x!=)
  205. throw ;
  206. vector<pair<pair<int,int>,int> > lst;
  207. pair<qword,qword> pr,res;
  208. for (int i=;i<totf;i++)
  209. {
  210. init_fact(factor[i][],factor[i][]);
  211. res=solve2(n,factor[i][],factor[i][],factor[i][]);
  212. pr=solve2(m,factor[i][],factor[i][],factor[i][]);
  213. pr.first=pow_mod(pr.first,factor[i][]-,factor[i][]);pr.second=-pr.second;
  214. res.first=res.first*pr.first%factor[i][];
  215. res.second=res.second+pr.second;
  216. pr=solve2(n-m,factor[i][],factor[i][],factor[i][]);
  217. pr.first=pow_mod(pr.first,factor[i][]-,factor[i][]);pr.second=-pr.second;
  218. res.first=res.first*pr.first%factor[i][];
  219. res.second=res.second+pr.second;
  220. res.first=res.first*pow_mod(factor[i][],res.second,factor[i][])%factor[i][];
  221. lst.push_back(make_pair(make_pair(factor[i][],factor[i][]),res.first));
  222. }
  223. qword ans;
  224. ans=solve4(lst);
  225. printf("%d\n",(int)ans);
  226. }
  227.  
  228. int main()
  229. {
  230. freopen("input.txt","r",stdin);
  231. int n,m,x,y,z;
  232. scanf("%d",&n);
  233. int opt;
  234. init();
  235. for (int i=;i<=n;i++)
  236. {
  237. scanf("%d%d%d%d",&opt,&x,&y,&z);
  238. if (opt==)
  239. {
  240. printf("%d\n",(int)pow_mod(x,y,z));
  241. }else if (opt==)
  242. {
  243. BabyStepGiantStep(x,y,z);
  244. }else if (opt==)
  245. {
  246. solve3(y,x,z);
  247. }
  248. }
  249. }

bzoj 3283: 运算器 扩展Baby Step Giant Step && 快速阶乘的更多相关文章

  1. [置顶] hdu2815 扩展Baby step,Giant step入门

    题意:求满足a^x=b(mod n)的最小的整数x. 分析:很多地方写到n是素数的时候可以用Baby step,Giant step, 其实研究过Baby step,Giant step算法以后,你会 ...

  2. HDU 2815 扩展baby step giant step 算法

    题目大意就是求 a^x = b(mod c) 中的x 用一般的baby step giant step 算法会超时 这里参考的是http://hi.baidu.com/aekdycoin/item/2 ...

  3. 【学习笔记】Baby Step Giant Step算法及其扩展

    1. 引入 Baby Step Giant Step算法(简称BSGS),用于求解形如\(a^x\equiv b\pmod p\)(\(a,b,p\in \mathbb{N}\))的同余方程,即著名的 ...

  4. 解高次同余方程 (A^x=B(mod C),0<=x<C)Baby Step Giant Step算法

    先给出我所参考的两个链接: http://hi.baidu.com/aekdycoin/item/236937318413c680c2cf29d4 (AC神,数论帝  扩展Baby Step Gian ...

  5. HDU 2815 Mod Tree 离散对数 扩张Baby Step Giant Step算法

    联系:http://acm.hdu.edu.cn/showproblem.php?pid=2815 意甲冠军: watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQ ...

  6. 『高次同余方程 Baby Step Giant Step算法』

    高次同余方程 一般来说,高次同余方程分\(a^x \equiv b(mod\ p)\)和\(x^a \equiv b(mod\ p)\)两种,其中后者的难度较大,本片博客仅将介绍第一类方程的解决方法. ...

  7. POJ 3243 Clever Y (求解高次同余方程A^x=B(mod C) Baby Step Giant Step算法)

    不理解Baby Step Giant Step算法,请戳: http://www.cnblogs.com/chenxiwenruo/p/3554885.html #include <iostre ...

  8. 数论之高次同余方程(Baby Step Giant Step + 拓展BSGS)

    什么叫高次同余方程?说白了就是解决这样一个问题: A^x=B(mod C),求最小的x值. baby step giant step算法 题目条件:C是素数(事实上,A与C互质就可以.为什么?在BSG ...

  9. 【POJ2417】baby step giant step

    最近在学习数论,然而发现之前学的baby step giant step又忘了,于是去翻了翻以前的代码,又复习了一下. 觉得总是忘记是因为没有彻底理解啊. 注意baby step giant step ...

随机推荐

  1. ant来历

    看看<ant权威指南>或者ant的网站吧,里面介绍得非常详细ant的来历,目的.ant是tomcat的一个副产品,是在开发tomcat过程中,开发人员需要一个整合编译工具,因为在一个大项目 ...

  2. 触摸与手势学习-swift

    触摸是一个UITouch对象,该对象放在一个UIEvent中,每个UIEvent包含一个或多个UITouch对象,每个UITouch对象对应一个手指.系统将UIEvent发送到应用程序上,最后应用程序 ...

  3. 解决ubuntu16.04下的sublime text3不能使用Fcitx下的搜狗输入法的问题

    Sublime Text 2/3 输入法(Fcitx)修复[Ubuntu(Debian)] 主要目的 安装 Sublime Text 3 安装 Fcitx 输入法 + 皮肤 修复 Sublime Te ...

  4. Mysql 数据库 操作语句

    mysql 格式语句规范 如何登陆你的数据库? 举例! 如果你的是 编译安装的花 那就得去编译安装后的那个目录中去,我的是安装到/usr/local/mysql 下登陆数据库:cd /usr/loca ...

  5. c语言,strcspn,在串中查找第一个给定字符集内容的段

    函数名: strcspn 功 能: 在串中查找第一个给定字符集内容的段 用 法: int strcspn(char *str1, char *str2); 程序例: #include <stdi ...

  6. 详说C#中的结构struct

    一.结构和类的区别 1.结构的级别和类一致,写在命名空间下面,可以定义字段.属性.方法.构造方法也可以通过关键字new创建对象. 2.结构中的字段不能赋初始值. 3.无参数的构造函数无论如何C#编译器 ...

  7. A题笔记(14)

    Reverse Words in a String : http://oj.leetcode.com/problems/reverse-words-in-a-string/ 代码 : https:// ...

  8. 经历:easyui的datagrid没有数据滚动条的显示

    今天,一个用户提出一个这样的问题,"查询不到结果时,为什么我看不到后面的标题呢?" 最初,我听到这个问题时,第一反应是:查出来数据不就有滚动条了吗,干嘛非要较真呢? 不过,后来想想 ...

  9. OC加强-day03

    #program mark - 0_18 分类的使用注意 [掌握] 1.分类的作用 作用:讲一个类分为多个模块,将相似功能的方法写在同一个模块中,方便我们后面代码的维护 "强调 1.分类中只 ...

  10. excel导入 导出 兼容各个版本服务器不装EXCEL也可以

    给出 demo源码: http://pan.baidu.com/s/1hqGMudY 提取码:pw4n首先要引用 NPOI.dll (可在网上下载!) //导入 public void OnSubmi ...