【BZOJ3122】随机数生成器(BSGS,数论)

232 篇文章 0 订阅
33 篇文章 0 订阅

题面

BZOJ
洛谷
这里写图片描述

题解

考虑一下递推式
发现一定可以写成一个
Xi+1=(X1+c)aic 的形式
直接暴力解一下
Xi+1+c=a(Xi+c)
解得 c=ba1
这样子,相当于得到了一个 kaxt+c(mod p) 这样的式子
这个显然是个裸的 BSGS
直接解出来就行了

注意特判一下 a=0,a=1,X1=t 这几种情况。

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<set>
#include<map>
#include<vector>
#include<queue>
using namespace std;
#define ll long long
#define RG register
inline int read()
{
    RG int x=0,t=1;RG char ch=getchar();
    while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
    if(ch=='-')t=-1,ch=getchar();
    while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
    return x*t;
}
int fpow(int a,int b,int MOD)
{
    int s=1;
    while(b){if(b&1)s=1ll*s*a%MOD;a=1ll*a*a%MOD;b>>=1;}
    return s;
}
const int HashMod=111111;
struct HashTable
{
    struct Line{int u,v,next;}e[100000];
    int h[HashMod],cnt;
    void Add(int u,int v,int w){e[++cnt]=(Line){w,v,h[u]};h[u]=cnt;}
    void clear(){memset(h,0,sizeof(h));cnt=0;}
    void Insert(int x,int i){Add(x%HashMod,i,x);}
    int Query(int x)
    {
        for(RG int i=h[x%HashMod];i;i=e[i].next)
            if(e[i].u==x)return e[i].v;
        return -1;
    }
}Hash;
int BSGS(int a,int y,int z,int p)
{
    if(y%p==0)return -2;
    if(z==a)return -1;
    Hash.clear();
    int m=sqrt(p)+1;
    for(RG int i=0,t=z;i<m;++i,t=1ll*t*y%p)Hash.Insert(t,i);
    for(RG int i=1,tt=fpow(y,m,p),t=1ll*a*tt%p;i<=m;++i,t=1ll*t*tt%p)
    {
        int B=Hash.Query(t);if(B==-1)continue;
        return i*m-B;
    }
    return -2;
}
int main()
{
    int T=read(),p,a,b,c,X1,t;
    while(T--)
    {
        p=read();a=read();b=read();X1=read();t=read();
        if(X1==t){puts("1");continue;}
        if(a==0){if(b==t)puts("2");else puts("-1");continue;}
        if(a==1)
        {
            if(b==0){puts("-1");continue;}
            t=(t+p-X1)%p;
            t=1ll*t*fpow(b,p-2,p)%p;
            printf("%d\n",t+1);
            continue;
        }
        c=1ll*b*fpow(a-1,p-2,p)%p;
        t=(t+c)%p;X1=(X1+c)%p;
        printf("%d\n",BSGS(X1,a,t,p)+1);
    }
    return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值