题目链接

题意:

  • 给定\(n,x_0,a,b,p\),有递推式\(x_i = (a \cdot x_{i-1} +b)\%p\)。
  • 有\(Q\)个询问,每次询问给定一个\(v\),问是否存在一个最小的\(i\)使得\(x_i=v,i\in[0,n-1]\)成立
  • \(1\le n\le 1e18,0\le x_p,a,b<p\le 1e9+7,Q\le 1000,p\)是质数

大步小步(BSGS)学习:https://oi-wiki.org/math/bsgs/

刘汝佳白书数论中也有入门讲解

如果直接看递推式找不到关系可以先列出前几项

\(x_0 = x_0\quad x_1=ax_0+b\quad x_2=a^2x_0+ab+b\quad x_3=a^3x_0+a^2b+ab+b\)

\(x_4=a^4x_0+\sum_{i=0}^3a^ib\quad x_5=a^5x_0+\sum_{i=0}^4a^ib\quad x_6=a^6x_0+\sum_{i=0}^5a^ib\quad x_7=a^7x_0+\sum_{i=0}^6a^ib\)

\(\cdots\)

在BSGS中,\(a^x\equiv b\pmod p\)的求解是前求出前\(m\)项(\(m\)通常取\(\sqrt p\))的答案(即\(a^i\%p,i\in[0,m]\)),存在一个数组中便于查询(值和位置一般都要存),然后再从\([m+1,2m]\)中找答案,如果这一层中有答案那么也就是说\(a^{i+m} \equiv b\pmod p,i+m\in [m+1,2m])\)可以发现如果把指数上的\(m\)挪到右边,也就是相当于右侧乘了一个\(a^m\)的逆元,式子变成了\(a^i\equiv b\cdot a^{-m}\pmod p\),所以我们在考虑第二层中是否有答案时,只需要将\(b\)先成一次\(a^m\)的逆元,然后再从我们预处理出来的第一层的答案中寻找是否有解即可。

那么这个题该怎么像上面这个进行一样的操作呢?

我们观察一下,考虑完上一层之后接着考虑下一层的时候,给b乘\(a^m\)的逆元操作意味着什么?该操作可以理解为是除以\(a^m\),也就是将乘了\(m\)次\(a\)的操作复原了。基于复原思想,如何复原到m次操作之前的结果呢?

回看之前列出来的前8项,假设一层有4个,那么复原就相当于是将\(x_4\)变回\(x_0\)。

\[a^4x_0+a^3b+a^2b+ab+b \rightarrow x_0
\]
\[{a^4x_0+a^3b+a^2b+ab+b \over a^4} = x_0+\frac ba +\frac b{a^2}+\frac b{a^3}+\frac b{a^4}
\]

如果还看不出来,在列一下下一个

\[a^5x_0+a^4b+a^3b+a^2b+ab+b \rightarrow ax_0+b
\]
\[{a^5x_0+a^4b+a^3b+a^2b+ab+b \over a^4} = (ax_0+b)+\frac ba +\frac b{a^2}+\frac b{a^3}+\frac b{a^4}
\]

显然,先除以\(a^4\),然后再减去\(\frac ba +\frac b{a^2}+\frac b{a^3}+\frac b{a^4}\)即可。

其他处理就几乎和BSGS入门题一样了。

该题如果每次查询\(O(\sqrt p)\)的话会超时,所以我们通过加大预处理的范围去降低时间复杂度。(更多的部分是可以直接算出来存到数组里面的)采取\(O(\sqrt {Qp})\)也就是1e6的大小。

代码&程序流程梳理

  1. 特判a == 0,如果查询vx0要先输出0,等于b再输出1,否则输出-1(注意一下如果x0b,答案要首先输出0)
  2. 预处理前1e6的答案,存在pair里面之后排序,去重
  3. 计算\(a^{1e6}\)的逆元,还有我们复原需要减去的那一坨东西
  4. 然后从第一层开始找,二分查询位置,如果没找到就找下一层,如果发现现在找的位置大于等于n的话要及时结束查找。
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxn = 1000000;
int T;
ll x0,n,a,b,p;
ll ksm(ll a,ll b){
ll res = 1;
for(;b;b>>=1){
if(b & 1)res = res*a%p;
a = a * a % p;
}
return res;
}
pair<int,int> node[maxn];
int pos[maxn],val[maxn];
int main(){
scanf("%d",&T);
while(T--){
scanf("%lld%lld%lld%lld%lld",&n,&x0,&a,&b,&p);
int Q;scanf("%d",&Q);
if(a == 0){
while(Q--){
int v;scanf("%d",&v);
if(v == x0)
puts("0");
else if(v == b) puts("1");
else puts("-1");
}
continue;
}
ll now = x0;
int up = min(n,(ll)maxn);
for(int i=0;i<up;i++){
node[i] = {now,i};
now = (now * a + b) % p;
}
sort(node,node+up);
int m = 0;
for(int i=0;i<up;i++){
val[m] = node[i].first;
pos[m++] = node[i].second;
while(i < up - 1 && node[i].first == node[i+1].first)i++;
}
int inv_a = ksm(a,p-2);
int inv_b = (p-b)%p*inv_a%p;
ll bb = 0, aa = 1;
for(int i=0;i<maxn;i++){
aa = aa * inv_a % p;
bb = (bb * inv_a + inv_b) % p;
}
int r = p / maxn + 1;
while(Q--){
int v;scanf("%d",&v);
int id = lower_bound(val,val+m,v) - val;
if(id < m && val[id] == v){
printf("%d\n",pos[id]);continue;
}
if(n < maxn){
puts("-1");continue;
}
bool flag = false;
for(int i=1;i<=r;i++){
v = (aa * v + bb) % p;
int id = lower_bound(val,val+m,v) - val;
if(id < m && val[id] == v){
int res = pos[id] + i * maxn;
if(res >= n)puts("-1");
else printf("%d\n",res);
flag = true;break;
}
}
if(!flag)puts("-1");
}
}
return 0;
}

最后提一下,复原的时候也可以先减后乘逆元

那么循环里面bb的更新就变成

ll bb = 0, aa = 1;
for(int i=0;i<maxn;i++){
aa = aa * inv_a % p;
bb = (bb*a%p + b) % p;
}
//循环查找中v的更新
v = (v-bb+p)%p*aa%p;

2019牛客暑期多校训练营(第五场)C - generator 2 (BSGS)的更多相关文章

  1. 2019牛客暑期多校训练营(第五场) maximum clique 1

    题意:给出n个不相同的数,问选出尽量多的数且任两个数字二进制下不同位数大于等于2. 解法:能想到大于等于2反向思考的话,不难发现这是一个二分图,那么根据原图的最大团等于补图的最大独立点集,此问题就变成 ...

  2. 2019牛客暑期多校训练营(第九场)A:Power of Fibonacci(斐波拉契幂次和)

    题意:求Σfi^m%p. zoj上p是1e9+7,牛客是1e9:  对于这两个,分别有不同的做法. 前者利用公式,公式里面有sqrt(5),我们只需要二次剩余求即可.     后者mod=1e9,5才 ...

  3. 2019牛客暑期多校训练营(第一场)A题【单调栈】(补题)

    链接:https://ac.nowcoder.com/acm/contest/881/A来源:牛客网 题目描述 Two arrays u and v each with m distinct elem ...

  4. 2019牛客暑期多校训练营(第一场) B Integration (数学)

    链接:https://ac.nowcoder.com/acm/contest/881/B 来源:牛客网 Integration 时间限制:C/C++ 2秒,其他语言4秒 空间限制:C/C++ 5242 ...

  5. 2019牛客暑期多校训练营(第一场) A Equivalent Prefixes ( st 表 + 二分+分治)

    链接:https://ac.nowcoder.com/acm/contest/881/A 来源:牛客网 Equivalent Prefixes 时间限制:C/C++ 2秒,其他语言4秒 空间限制:C/ ...

  6. 2019牛客暑期多校训练营(第二场)F.Partition problem

    链接:https://ac.nowcoder.com/acm/contest/882/F来源:牛客网 Given 2N people, you need to assign each of them ...

  7. 2019牛客暑期多校训练营(第一场)A Equivalent Prefixes(单调栈/二分+分治)

    链接:https://ac.nowcoder.com/acm/contest/881/A来源:牛客网 Two arrays u and v each with m distinct elements ...

  8. [状态压缩,折半搜索] 2019牛客暑期多校训练营(第九场)Knapsack Cryptosystem

    链接:https://ac.nowcoder.com/acm/contest/889/D来源:牛客网 时间限制:C/C++ 2秒,其他语言4秒 空间限制:C/C++ 262144K,其他语言52428 ...

  9. 2019牛客暑期多校训练营(第二场)J-Subarray(思维)

    >传送门< 前言 这题我前前后后看了三遍,每次都是把网上相关的博客和通过代码认真看了再思考,然并卵,最后终于第三遍也就是现在终于看懂了,其实懂了之后发现其实没有那么难,但是的的确确需要思维 ...

  10. 2019牛客暑期多校训练营(第一场)-A (单调栈)

    题目链接:https://ac.nowcoder.com/acm/contest/881/A 题意:给定两个长度均为n的数组a和b,求最大的p使得(a1,ap)和(b1,bp)等价,等价的定义为其任意 ...

随机推荐

  1. flask socketio 踩坑记录

    在使用python3的flask-socketio+socket.io.js的时候报错 在使用python3的flask-socketio+socket.io.js的时候报错"unsuppo ...

  2. kafka 异步双活方案 mirror maker2 深度解析

    mirror maker2背景 通常情况下,我们都是使用一套kafka集群处理业务.但有些情况需要使用另一套kafka集群来进行数据同步和备份.在kafka早先版本的时候,kafka针对这种场景就有推 ...

  3. springboot异常处理之404

    ps: 推荐一下本人的通用后台管理项目crowd-admin 以及newbee-mall增强版,喜欢的话给个star就好 源码分析 在springboot中默认有一个异常处理器接口ErrorConto ...

  4. 【SpringBoot】Spring Boot,开发社区讨论交流网站首页。

    初识Spring Boot,开发社区讨论交流网站首页. 文章目录 初识Spring Boot,开发社区讨论交流网站首页. 1.项目简介 2. 搭建开发环境 JDK Apache Maven Intel ...

  5. 计网Q1:多个方面比较电路交换、报文交换和分组交换的主要优缺点

    网上看到的带佬儿的帖子......膜过来<doge 原文链接: https://blog.csdn.net/njchenyi/article/details/1540657 电路交换: 由于电路 ...

  6. ctfshow—web—web7

    打开靶机 发现是SQL注入,盲注 过滤了空格符,可以用/**/绕过,抓包 直接上脚本 import requestss=requests.session()url='https://46a0f98e- ...

  7. 快速查询表中的NULL数据

    正常情况下,NULL值是不会放入B-TREE索引的,因此根据IS NULL查询的时候走的通常是全表扫描,如果记录比较少还好,记录比较多,查询会非常耗时 可以通过创建一个索引来解决 CREATE IND ...

  8. C++ STL 优先队列 (priority_queue)

    std::priority_queue <queue> 优先队列   优先队列是一种容器适配器,根据某些严格的弱排序标准,使其第一个元素始终包含的最大元素.   这种特性类似于堆,它可以在 ...

  9. Java自学笔记1206

    字符串比较string1.equals(string2) 代码如下: 1 package Demo_1206; 2 3 import java.util.Scanner; 4 5 public cla ...

  10. windows中使用django时报错:A server error occurred. Please contact the administrator.

    这是因为在视图函数中使用了get函数,获取了不存在的数据例如:数据库中不存在一条name为hello1的数据,使用如下语句访问message = Message.objects.get(name='h ...