「学习笔记」二次剩余 - 二次剩余 - 学习笔记

本文探讨了在模质数意义下求解二次剩余问题的算法,通过随机选取整数并利用特定数学性质,实现了O(lg p)的时间复杂度求解。文章详细介绍了算法步骤,包括判定二次剩余的存在性和求解过程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

下文只在模质数意义下讨论。
即给定n>0,p,pn>0,p,pn>0,p,p是质数,求所有x∈[0,p),x2=n(modp)x\in[0,p),x^2=n\pmod px[0,p),x2=n(modp)
若存在解则nnn称为ppp的二次剩余。
首先将p=2p=2p=2的特殊情况判掉。下文ppp是奇质数。
首先显然模奇质数ppp意义下会有恰好p−12\frac{p-1}22p1个二次剩余。
首先如何判定,有一个定理:
(−1)[n不是p的二次剩余]=np−12(modp)(-1)^{[n不是p的二次剩余]}=n^{\frac{p-1}2}\pmod p(1)[np]=n2p1(modp)
并且若有解xxx,则仅有xxxp−xp-xpx这两组解。这个定理易证,略去证明。
(上述两个定理都可以通过百度“二次剩余”搜到证明(应该能))

那么考虑一个算法:在[1,p)[1,p)[1,p)随机一个整数ttt,使得:t2−nt^2-nt2n不是关于ppp的二次剩余。由于有恰好一半不是,因此随机两步就会随机出一组。那么令w=t2−nw=\sqrt{t^2-n}w=t2n(这里可能不存在在模ppp意义下后半部分的平方根,这里只是定义一个有w2=t2−n(modp)w^2=t^2-n\pmod pw2=t2n(modp)性质的www)。
可以显然证明:wP=−ww^P=-wwP=w
那么考虑令x=(t+w)p+12x=(t+w)^{\frac{p+1}2}x=(t+w)2p+1
x2=(t+w)p+1=(t+w)(t+w)p=(t+w)∑i=0p(pi)tiwp−i=(t+w)(tp+wp)=(t+w)(t−w)=t2−w2=t2−t2+n=nx^2=(t+w)^{p+1}\\=(t+w)(t+w)^p\\=(t+w)\sum_{i=0}^p\binom pit^iw^{p-i}\\=(t+w)(t^p+w^p)\\=(t+w)(t-w)=t^2-w^2\\=t^2-t^2+n=nx2=(t+w)p+1=(t+w)(t+w)p=(t+w)i=0p(ip)tiwpi=(t+w)(tp+wp)=(t+w)(tw)=t2w2=t2t2+n=n
因此xxx就是x2=n(modp)x^2=n\pmod px2=n(modp)的一组解,其解集为{x,p−x}\{x,p-x\}{x,px}
时间复杂度O(lg⁡p)O(\lg p)O(lgp)。实现的时候可以将(t+w)k(t+w)^k(t+w)k视为v+cwv+cwv+cw,利用w2=t2−nw^2=t^2-nw2=t2n即可做一次乘法。代码(应该是正确的):

#include<bits/stdc++.h>
#define rep(i,a,b) for(int i=a;i<=b;i++)
#define gc getchar()
using namespace std;typedef long long lint;
inline int inn() { int x,ch;while((ch=gc)<'0'||ch>'9');x=ch^'0';while((ch=gc)>='0'&&ch<='9') x=(x<<1)+(x<<3)+(ch^'0');return x; }
inline int fast_pow(int x,int k,int p,int ans=1) { for(;k;k>>=1,x=(lint)x*x%p) (k&1)?ans=(lint)ans*x%p:0;return ans; }
struct RAND{ unsigned int x;RAND() { x=1000000007; }
	inline int operator() (int a,int b) { return x=(x<<11)^(x>>3)^x^998244353,int(x%(b-a+1)+a); }
}rnd; struct E{ int t,c;E(int _t=0,int _c=0) { t=_t,c=_c; } };
inline E tms(const E &a,const E &b,int w2,int p) { return E(((lint)a.t*b.t+(lint)a.c*b.c%p*w2)%p,((lint)a.c*b.t+(lint)a.t*b.c)%p); }
inline int calc(E x,int w2,int k,int p) { E ans(1,0);for(;k;k>>=1,x=tms(x,x,w2,p)) if(k&1) ans=tms(ans,x,w2,p);return ans.t; }
inline int solve(int n,int p)
{
	if(fast_pow(n,(p-1)/2,p)==p-1) return -1;int t,w;
	do{
		t=rnd(1,p-1),w=((lint)t*t-n)%p;if(w<0) w+=p;
	}while(fast_pow(w,(p-1)/2,p)==1);
	return calc(E(t,1),w,(p+1)/2,p);
}
int main()
{
	for(int T=inn();T;T--)
	{
		int n=inn(),p=inn();n%=p;if(!n) { printf("0\n");continue; }
		if(p==2) { printf("1\n");continue; }int ans1=solve(n,p),ans2=p-ans1;
		if(ans1==-1) { printf("No answer!\n");continue; }
		if(ans1>ans2) swap(ans1,ans2);printf("%d %d\n",ans1,ans2);
	}
	return 0;
}
`
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值