UOJ 74 [UR #6]破解密码

逆元

就说这种题怎么考场上没有人AC,原来有坑点- -

考虑 hi hi+1 的关系:
hi+1hi2626nsi+si(mod p)
(126n)sihi+126hi(mod p)
sihi+126hi126n(mod p)
因为有解,且模数 p>26 ,因此我们似乎可以一位一位地确定答案。
然而坑点来了,可能出现 126n|p ,此时它并没有逆元,解不唯一,将会表现出 az 都合法的情况,就会挂掉。为什么?从②式可以看出并不是没逆元的锅,只是因为我们列出的式子只能说明在这个取值下 hi hi+1 可以同时合法,而不能保证 hi 和其他的能同时合法。这样是对的当且仅当存在唯一解 (因为不可能有其他可行解)。而在这个坑点时没有唯一解。怎么办?
回到①式,发现满足 126n|p 时一定会出现:
hi+1hi26(mod p)
又因为保证有解,可知如果满足了 hi hi+1 一定也满足,因此暴力地让 h0 满足即可。

#include<cstdio>
#define N 100005
using namespace std;
namespace runzhe2000
{
    typedef long long ll;
    int n, p, y[N];
    int fpow(int a, int b)
    {
        int r = 1; a = (a%p+p)%p;
        for(; b; b>>=1)
        {
            if(b&1)r=(ll)r*a%p;
            a=(ll)a*a%p;
        }
        return r;
    }
    void main()
    {
        scanf("%d%d",&n,&p);
        for(int i = 1; i <= n; i++) scanf("%d",&y[i]);
        if(fpow(26,n) == 1)
        {
            int tmp = y[1];
            for(int i = n; i; i--)
            {
                int c = tmp % 26;
                y[i] = c;
                tmp /= 26;
            }
            for(int i = 1; i <= n; i++) printf("%c",y[i]+'a');
        }
        else
        {
            int inv = fpow(fpow(26,n)-1, p - 2);
            y[n+1] = y[1];
            for(int i = 1; i <= n; i++)
            {
                int a = (ll) (y[i] * 26ll - y[i+1]) % p * inv % p;
                if(a < 0) a += p;
                printf("%c",a+'a');
            }
        } 
    }
}
int main()
{
    runzhe2000::main();
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值