题目

几乎原题 BZOJ3122题解

分析

先推一波公式,然后除去特殊情况分类讨论,剩下就是形如 $a^i \equiv b(mod \ p)$ 的方程,可以使用BSGS算法。

在标准的BSGS中,内外层循环都是 $\sqrt p$,题目查询 $m$ 次,$m \leq 1000$,$ p \leq 10^9$,这样总时间复杂度为 $O(m \sqrt p)$,勉强能接受。据说使用读入优化和手写哈希还是能过得,可见Cls的代码%%%

仔细想一下,由于BSGS分成两步,其中第一步是建立 $a$ 的幂次方的表,而题恰好是 $a,p$相同下的一组询问,所以这一部分可以与处理。

显然,我们应该对这部分分大点,第二步就会小写(因为积一定),

如分成 $\sqrt {mp} \times \sqrt{\frac{p}{m}}$(前面是预处理时间,后面是每次查询的时间)

也可以是 $\frac{p}{1000} \times 1000$,总之,要是预处理部分大一点。

  1. #include<bits/stdc++.h>
  2. #define rep(i,a,n) for(int i=a;i<=n;i++)
  3. #define per(i,a,n) for(int i=n;i>=a;i--)
  4. #define pb push_back
  5. #define mp make_pair
  6. #define FI first
  7. #define SE second
  8. #define maxn 100000
  9. #define inf 0x3f3f3f3f
  10. using namespace std;
  11. typedef long long ll;
  12. typedef unsigned long long ull;
  13. typedef pair<int,int> pii;
  14. typedef vector<int> vi;
  15. typedef double db;
  16.  
  17. int mod;
  18. int m=;
  19.  
  20. const int M=;
  21. int hs[M],head[M],nxt[M],id[M],top;
  22. void insert(int x,int y)
  23. {
  24. int k=x%M;
  25. hs[top]=x,id[top]=y,nxt[top]=head[k],head[k]=top++;
  26. }
  27. int find(int x)
  28. {
  29. int k=x%M;
  30. for(int i=head[k];i!=-;i=nxt[i]) if(hs[i]==x) return id[i];
  31. return -;
  32. }
  33.  
  34. int BSGS(int a,int b,int n)
  35. {
  36. if(b==) return ;
  37. ll p=;
  38. int ans=inf;
  39. rep(i,,m)
  40. {
  41. int id=find(1ll*p*b%n);
  42. if(id!=-) ans=min(ans,id-i);
  43. p=p*a%mod;
  44. }
  45. if(ans==inf) return -;
  46. return ans;
  47. }
  48.  
  49. ll qp(ll a,ll k)
  50. {
  51. ll res=;
  52. while(k)
  53. {
  54. if(k&) res=res*a%mod;
  55. a=a*a%mod;
  56. k>>=;
  57. }
  58. return res;
  59. }
  60. ll _inv(ll x) {return qp(x,mod-);}
  61.  
  62. int main()
  63. {
  64. int CAS; scanf("%d",&CAS);
  65. while(CAS--)
  66. {
  67. ll N; int x0,a,b; scanf("%lld%d%d%d%d",&N,&x0,&a,&b,&mod);
  68. memset(head,-,sizeof head); top=;
  69.  
  70. int bn=(mod+m-)/m;
  71. int aA=qp(a,m),pw=aA;
  72. rep(i,,bn)
  73. {
  74. if(find(pw)==-) insert(pw,i*m);
  75. pw=1ll*pw*aA%mod;
  76. }
  77. int Q; scanf("%d",&Q);
  78. while(Q--)
  79. {
  80. int v; scanf("%d",&v);
  81. if(a==)
  82. {
  83. if(v==x0) puts("");
  84. else if(v==b)
  85. {
  86. if(N==) puts("-1");
  87. else puts("");
  88. }
  89. else puts("-1");
  90. }
  91. else if(a==)
  92. {
  93. if(b==)
  94. {
  95. if(v==x0) puts("");
  96. else puts("-1");
  97. }
  98. else
  99. {
  100. ll n=1ll*(v+mod-x0)*_inv(b)%mod;
  101. if(n>=N) puts("-1");
  102. else printf("%lld\n",n);
  103. }
  104. }
  105. else
  106. {
  107. v=(1ll*(a-)*v+b)%mod;
  108. int X=(1ll*(a-)*x0+b)%mod;
  109. if(X==)
  110. {
  111. if(v==) puts("");
  112. else puts("-1");
  113. }
  114. else
  115. {
  116. v=1ll*v*_inv(X)%mod;
  117. if(v==) puts("-1");
  118. else
  119. {
  120. int n=BSGS(a,v,mod);
  121. if(n>=N) puts("-1");
  122. else printf("%d\n",n);
  123. }
  124. }
  125. }
  126. }
  127. }
  128. return ;
  129. }

参考链接:https://ac.nowcoder.com/acm/contest/view-submission?submissionId=41008097

2019牛客多校第五场generator2——BSGS&&手写Hash的更多相关文章

  1. 2019牛客多校第五场 generator 1——广义斐波那契循环节&&矩阵快速幂

    理论部分 二次剩余 在数论中,整数 $X$ 对整数 $p$ 的二次剩余是指 $X^2$ 除以 $p$ 的余数. 当存在某个 $X$,使得式子 $X^2 \equiv d(mod \ p)$ 成立时,称 ...

  2. 2019牛客多校第五场F maximum clique 1 最大独立集

    题意:给你n个数,现在让你选择一个数目最大的集合,使得集合中任意两个数的二进制表示至少有两位不同,问这个集合最大是多大?并且输出具体方案.保证n个数互不相同. 思路:容易发现,如果两个数不能同时在集合 ...

  3. 2019牛客多校第五场G-subsequence 1 DP

    G-subsequence 1 题意 给你两个字符串\(s.t\),问\(s\)中有多少个子序列能大于\(t\). 思路 令\(len1\)为\(s\)的子序列的长度,\(lent\)为\(t\)的长 ...

  4. 2019牛客多校第五场 B - generator 1 矩阵快速幂+十倍增+二进制倍增优化

    B - generator 1 题意 给你\(x_{0}.x_{1}.a.b.b.mod\),根据\(x_{i} = a*x_{i-1} + b*x_{i-2}\)求出\(x_{n}\) 思路 一般看 ...

  5. 2019牛客多校第五场H - subsequence 2 拓扑

    H - subsequence 2 题意 要你使用前\(m\)个小写字母构造一个长度为\(n\)的字符串 有\(m*(m-1)/2\)个限制条件: \(c_{1} .c_{2}. len\):表示除去 ...

  6. 2019牛客多校第五场B-generator 1(矩阵快速幂)

    generator 1 题目传送门 解题思路 矩阵快速幂.只是平时的矩阵快速幂是二进制的,这题要用十进制的快速幂. 代码如下 #include <bits/stdc++.h> #defin ...

  7. 2019 牛客多校第五场 B generator 1

    题目链接:https://ac.nowcoder.com/acm/contest/885/B 题目大意 略. 分析 十进制矩阵快速幂. 代码如下 #include <bits/stdc++.h& ...

  8. 2019牛客多校第五场 G subsequence 1 dp+组合数学

    subsequence 1 题意 给出两个数字串s,t,求s的子序列中在数值上大于t串的数量 分析 数字大于另一个数字,要么位数多,要么位数相同,字典序大,位数多可以很方便地用组合数学来解决,所以只剩 ...

  9. 2019牛客多校第五场 F maximum clique 1 状压dp+最大独立集

    maximum clique 1 题意 给出一个集合s,求每个子集的最大独立集的权值和(权值是独立集的点个数) 分析 n比较小,一股浓浓的暴力枚举每一个子集的感觉,但是暴力枚举模拟肯定会T,那么想一想 ...

随机推荐

  1. 使用网关zuul过滤器登录鉴权

    使用网关zuul过滤器登录鉴权     1.新建一个filter包         filte有很多种 pre.post.     2.新建一个类LoginFilter,实现ZuulFilter,重写 ...

  2. Python使用datetime来判断近七天

    目录 strptime 使用strptime来格式化字符串 datetime.datetime.strptime("2019-10-02", "%Y-%m-%d" ...

  3. 【规律】Parentheses

    Parentheses 题目描述 Dave loves strings consisting only of '(' and ')'. Especially, he is interested in ...

  4. PHP对程序员的要求更高

     我这个文章标题可不是和大家开玩笑的哦  首先, 大家都知道, PHP也是一种编译型脚本语言, 和其他的预编译型语言不同, 它不是编译成中间代码, 然后发布.. 而是每次运行都需要编译.. 为此, 也 ...

  5. html如何点击子元素事件而不触发父元素的点击事件——阻止冒泡

    如果子元素和父元素都有点击事件,会出现点击事件冒泡的情况. 1.如何避免冒泡: html: <html> <head></head> <body> &l ...

  6. jvm类加载原理和类加载器介绍

    虚拟机的类加载机制 在Class文件中描述的各种信息最终都需要加载到虚拟机中之后才能运行和使用.   虚拟机把描述类的数据从Class文件加载到内存,并对数据进行校验,转换解析和初始化,最终形成可以被 ...

  7. Angular 调试

    我们新建一个项目.执行 ng server 会启动一个网站. 1. 执行 where ng .看看ng 是什么. D:\Abp学习\angular\Mytest>where ng C:\User ...

  8. [Vuex系列] - Actions的理解之我见

    Actions如何定义的 恕小端不才,对Action的总结如下: Action 可以提交mutation方法,通过mutation来改变state Action 函数可以接收一个context对象,通 ...

  9. vue 的虚拟 DOM 有什么好处?

    vue 中的虚拟DOM有什么好处?快! 首先了解浏览器显示网页经历的5个过程 1.解析标签,生成元素树(DOM树) 2.解析样式,生成样式树 3.生成元素与样式的关系 4.生成元素的显示坐标 5.显示 ...

  10. 用python实现新词发现程序——基于凝固度和自由度

    互联网时代,信息产生的数量和传递的速度非常快,语言文字也不断变化更新,新词层出不穷.一个好的新词发现程序对做NLP(自然预言处理)来说是非常重要的. N-Gram加词频 最原始的新词算法莫过于n-gr ...