zoukankan      html  css  js  c++  java
  • P4454 [CQOI2018]破解DH协议

    链接

    这题并不难只是需要把题读懂 — By ShadderLeave

    一句话题意

    给定两个数 \(p\)\(g\),有\(t\)组询问,每组询问给出\(A\)\(B\)

    其中 A = \(g^a \bmod p\) B = \(g^b \bmod p\)

    问你\(g^{ab} \bmod p\)是多少。

    初步解法就是用BSGS求出每个\(a\),\(b\)在用快速幂算出\(g^{ab} \bmod p\)

    可实际上你就会发现只要算一个就行。

    算出\(a\)直接求出\(B^a \bmod p\)就是答案

    然鹅,就这样交上去你就会狂TLE

    所以,我们只能再考虑优化。

    每次询问,我们都会把map清空,并重新储存,但这样会浪费很多时间,那我们从这开始优化

    我们要求的是这个柿子 \(g^a \equiv A\)

    我们利用BSGS的思想可以把它化为 \(g^{kt+b} \equiv A\)

    也就是\(g^{kt} \equiv A \times g^B\)

    发现方程右边会随A的取值发生变化,但左边的g和t确定了,那么值就不会变。

    所以,我们可以预先处理出\(g^{kt}\)并把他插入map中。

    对于每组询问,枚举\(A \times g^j\) 看在map中是否出现过。

    如果出现过,答案就是 map中的存的幂指数 - \(j\)

    但有一个很大的问题就是:

    卡 。。。常 。。。。

    卡。。。。。常。。。。

    卡。。。。。。常。。。。。

    毒瘤出题人nmsl

    所以我们只能少用快速幂,再求\(g^{kt}\)以及\(g^j\)只能用累乘的方法来求。

    出题人我*****

    代码

    
    #include<iostream>
    #include<cstdio>
    #include<algorithm>
    #include<map>
    #include<cmath>
    using namespace std;
    #define LL long long
    int g,t,p,A,B;
    map<LL,int> hash;
    inline LL read()
    {
    	LL s = 0, w = 1; char ch = getchar();
    	while(ch < '0' || ch > '9'){if(ch == '-') w = -1; ch = getchar();}
    	while(ch >= '0' && ch <= '9'){s= s * 10+ch - '0'; ch = getchar();}
    	return s * w;
    }
    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;
    }
    void YYCH()//预处理出g^kt
    {
        LL t = sqrt(p) + 1;
        LL base = ksm(g,t); LL tmp = 1;
        for(int i = 1; i <= t; i++)
        {
            tmp = tmp * base % p;//累乘避免被卡常
        	hash[tmp] = i * t;
        }
    }
    LL BSGS(int k)
    {
    	LL t = sqrt(p) + 1; LL tmp = k;
    	if(hash[tmp]) return hash[tmp];
    	for(int i = 1; i < t; i++)//枚举A*g^j
    	{
    		tmp = tmp * g % p;
    		if(hash[tmp]) return hash[tmp] - i;
    	}
    //	return -1;
    }
    int main()
    {
    	g = read(); p = read(); t = read(); YYCH();
    	while(t--)
    	{
    		A = read(); B = read();
    		printf("%lld\n",ksm(B,BSGS(A)));
    	}
    	return 0;
    }
    
    

    我拿出我珍藏多年的卡常火车头,出题人(17张牌你能秒杀我)你要是能卡住我,我当场把屏幕吃掉。

    呜呜,我错了,放过我吧,不要再卡我了。

  • 相关阅读:
    获取ip地址,
    手机div侧滑删除
    swiper左右选项卡滑动
    table-cell使用
    返回和刷新
    电脑浏览器计算高度和宽度
    css 空格
    时间js
    Nodejs仿Apache的部分功能
    Nodejs中的JavaScript
  • 原文地址:https://www.cnblogs.com/genshy/p/13507389.html
Copyright © 2011-2022 走看看