BSGS、EXBSGS

大步小步算法英文名: baby-step gaint-step (BSGS) .

该算法可在 Θ ( p ) \Theta \left ( \sqrt{p} \right ) Θ(p )的时间复杂度内求解
a x ≡ b   m o d   p a^{x} \equiv b\ mod\ p axb mod p,其中 a a a p p p互质,方程的解 x x x满足 0 ≤ x ≤   p 0 \leq x \leq\ p 0x p

x = A ⌈ p ⌉ − B x=A\left \lceil \sqrt{p} \right \rceil-B x=Ap B,其中 0 ≤ A , B ≤ ⌈ p ⌉ 0\leq A,B\leq \left \lceil \sqrt{p} \right \rceil 0A,Bp

则有 a A ⌈ p ⌉ − B ≡ b a^{A\left \lceil \sqrt{p} \right \rceil-B}\equiv b aAp Bb,即 a A ⌈ p ⌉ ≡ b a B a^{A\left \lceil \sqrt{p} \right \rceil}\equiv ba^{B} aAp baB

由于a、b是确定的,所以我们可以先枚举右边 b a B ba^{B} baB存入hash/map中,再枚举左边 a A ⌈ p ⌉ a^{A\left \lceil \sqrt{p} \right \rceil} aAp 的A,从而计算出所有的x

P3846 [TJOI2007] 可爱的质数/【模板】BSGS

代码如下

#include<bits/stdc++.h>

using namespace std;

typedef long long LL;

int main()
{
	int p, a, b;
	cin >> p >> a >> b;
	
	int m = (int)sqrt(p);
	if(m * m != p) m ++;
	
	map<int, int> hash;
	int s = 1;
	for(int i = 1; i <= m; ++ i)
	{
		s = (LL)s * a % p;
		b = (LL)b * a % p;
		hash[b] = i;
	}
	
	bool flag = false;
	a = s;
	for(int i = 1; i <= m; ++ i)
	{
		if(hash[a])
		{
			printf("%d\n", i * m - hash[a]);
			flag = true;
			break;
		}
		
		a = (LL)a * s % p;
	}
	
	if(!flag) printf("no solution");
	
	return 0;
}

扩展大步小步算法

上边的算法只针对 a a a p p p互质的情形,如果两者不互质,怎么计算?

x > 0 x>0 x>0时, a x ≡ b ( m o d   p ) a^{x}\equiv b(mod\ p) axb(mod p)等价于 a x − 1 a + n p = b a^{x-1}a+np=b ax1a+np=b裴蜀定理可知该方程有解的条件是 b b b g c d ( a , p ) gcd(a,p) gcd(a,p)的倍数

d = g c d ( a , p ) d=gcd(a,p) d=gcd(a,p),则有 a x − 1 a b + n p d = b d \frac{a^{x-1}a}{b}+\frac{np}{d}=\frac{b}{d} bax1a+dnp=db

a b a x − 1 ≡ b d ( m o d   p d ) \frac{a}{b}a^{x-1}\equiv \frac{b}{d}(mod\ \frac{p}{d}) baax1db(mod dp)

这时如果 g c d ( a , p d ) = 1 gcd(a,\frac{p}{d})=1 gcd(a,dp)=1,就可以直接使用BSGS求解,左侧多出一个系数,修改一下BGSG即可,解出的 x x x加上 1 1 1即是原方程的一个解,否则可以继续递归下去,直到互质为止

BSGS算出来的是最小解,而扩展BSGS算出来的不一定是最小解

注意:递归时需要特判,如果 b ≡ 1 b\equiv 1 b1直接返回 x = 0 x=0 x=0即可,当 a ≡ b ≡ 0 a\equiv b\equiv0 ab0时应该返回1,一般认为 0 0 = 1 0^0=1 00=1

时间复杂度: Θ ( n l o g n ) \Theta (\sqrt{n}logn) Θ(n logn)

P4195 【模板】扩展BSGS

具体代码如下

#include<bits/stdc++.h>

using namespace std;

typedef long long LL;

int gcd(int a, int b)
{
	return b ? gcd(b, a % b) : a;
}

map<int, int> hs;
int BSGS(int a, int b, int p, int k)
{
	hs.clear();
	int m = sqrt(p) + 1, s = 1;
	for(int B = 1; B <= m; ++ B)
	{
		s = 1ll * s * a % p;
		hs[1ll * b * s % p] = B;
	}
	
	k = 1ll * k * s % p;
	for(int i = 1; i <= m; ++ i)
	{
		int j = hs.count(k) ? hs[k] : -1;
		if(j >= 0 && i * m - j >= 0) return i * m - j;
		k = 1ll * k * s % p;
	}
	return -1;
}

int exBSGS(int a, int b, int p)
{
	if(b == 1 || p == 1) return 0;
	int d = gcd(a, p), k = 1, cnt = 0;
	while(d != 1)
	{
		if(b % d) return -1;
		cnt ++;
		b /= d, p /= d;
		k = (LL)k * a / d % p;
		if(k == b) return cnt;
		d = gcd(a, p); 
	}
	
	int res = BSGS(a, b, p, k);
	if(res < 0) return -1;
	return res + cnt;
}

int main()
{
	int a, b, p;
	while(~scanf("%d%d%d", &a, &p, &b), (a || p || b))
	{
		int x = exBSGS(a % p, b % p, p);
		if(x < 0) puts("No Solution");
		else printf("%d\n", x);
	}
	
	return 0;
}

上边exBSGS函数中有一句if(k == b) return cnt,转化为公式是

a c n t d c n t × a x − c n t ≡ b d c n t ( m o d   p d c n t ) \frac{a^{cnt}}{d^{cnt}}\times a^{x-cnt}\equiv \frac{b}{d^{cnt}}(mod\ \frac{p}{d^{cnt}}) dcntacnt×axcntdcntb(mod dcntp)

其中 a c n t d c n t \frac{a^{cnt}}{d^{cnt}} dcntacnt就是 k k k b d c n t \frac{b}{d^{cnt}} dcntb就是 b b b k = b k=b k=b时相当于

a c n t d c n t ≡ b d c n t ( m o d   p d c n t ) \frac{a^{cnt}}{d^{cnt}}\equiv\frac{b}{d^{cnt}}(mod\ \frac{p}{d^{cnt}}) dcntacntdcntb(mod dcntp)

这个等同于

a c n t ≡ b ( m o d   p ) a^{cnt}\equiv b(mod\ p) acntb(mod p)

此时cnt就是我们要求的 x x x,直接返回即可

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
基于微信小程序的家政服务预约系统采用PHP语言和微信小程序技术,数据库采用Mysql,运行软件为微信开发者工具。本系统实现了管理员和客户、员工三个角色的功能。管理员的功能为客户管理、员工管理、家政服务管理、服务预约管理、员工风采管理、客户需求管理、接单管理等。客户的功能为查看家政服务进行预约和发布自己的需求以及管理预约信息和接单信息等。员工可以查看预约信息和进行接单。本系统实现了网上预约家政服务的流程化管理,可以帮助工作人员的管理工作和帮助客户查询家政服务的相关信息,改变了客户找家政服务的方式,提高了预约家政服务的效率。 本系统是针对网上预约家政服务开发的工作管理系统,包括到所有的工作内容。可以使网上预约家政服务的工作合理化和流程化。本系统包括手机端设计和电脑端设计,有界面和数据库。本系统的使用角色分为管理员和客户、员工三个身份。管理员可以管理系统里的所有信息。员工可以发布服务信息和查询客户的需求进行接单。客户可以发布需求和预约家政服务以及管理预约信息、接单信息。 本功能可以实现家政服务信息的查询和删除,管理员添加家政服务信息功能填写正确的信息就可以实现家政服务信息的添加,点击家政服务信息管理功能可以看到基于微信小程序的家政服务预约系统里所有家政服务的信息,在添加家政服务信息的界面里需要填写标题信息,当信息填写不正确就会造成家政服务信息添加失败。员工风采信息可以使客户更好的了解员工。员工风采信息管理的流程为,管理员点击员工风采信息管理功能,查看员工风采信息,点击员工风采信息添加功能,输入员工风采信息然后点击提交按钮就可以完成员工风采信息的添加。客户需求信息关系着客户的家政服务预约,管理员可以查询和修改客户需求信息,还可以查看客户需求的添加时间。接单信息属于本系统里的核心数据,管理员可以对接单的信息进行查询。本功能设计的目的可以使家政服务进行及时的安排。管理员可以查询员工信息,可以进行修改删除。 客户可以查看自己的预约和修改自己的资料并发布需求以及管理接单信息等。 在首页里可以看到管理员添加和管理的信息,客户可以在首页里进行家政服务的预约和公司介绍信息的了解。 员工可以查询客户需求进行接单以及管理家政服务信息和留言信息、收藏信息等。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值