《洛谷深入浅出进阶篇》P5091 —— 扩展欧拉定理,快读取模。

P5091 【模板】扩展欧拉定理 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn)icon-default.png?t=N7T8https://www.luogu.com.cn/problem/P5091

大概题意:给出三个正整数,a,m,b 你需要求出 a^{b}mod\ m的值,其中a<10^{9},b<10^{20000000},m<10^{8}

首先求出m的欧拉函数\psi(m),由欧拉定理可知, 当b>\psi(m)时,a^{b}\equiv a^{b\ mod\ \psi(m)+\psi(m)}(mod\ m),

这样我们就可以把b缩小到2\psi(m)之内了。

首先我们需要读入b,由于b太大了,不可能一次性读完,这样longlong都会爆,所以我们要一个字

符一个字符读入,然后边读边取模即可,最后返回的一定是一个小于2\psi(m)的数,

然后对a^{2\psi(m)}这种数量级的数取模,这应该不用我多讲吧,快速幂即可。

第一步:求出m的欧拉函数

ll phi(ll n) {
	ll ans = n;
	for (ll i = 2; i<= n/i; i++) {
		if (n % i == 0) {
			ans = ans / i * (i - 1);
			while (n % i == 0)n /= i;
		}
	}
	if (n != 1)ans =( ans / n) * (n - 1);
	return ans;
}

第二步:边读边模求指数;

inline ll read(ll n) {
	ll f = 0,x=0;
	char c = getchar();

	while (!isdigit(c))c=getchar();

	while (isdigit(c)) {
		x = x * 10 + c-'0';
		if (x >= n)f = 1;
		x %= n;
		c = getchar();
	}
	return x+(f == 1 ?  n : 0);
}

第三步:快速幂

ll fast_pow(ll a,ll b,ll m) {
	ll s = 1;
	while (b) {
		if (b & 1)s = s * a % m;
		a = a * a % m;
		b >>= 1;
	}
	return s;
}

第四步:合并

#define _CRT_SECURE_NO_WARNINGS
#include<iostream>
#include<cstdio>
#include<cmath>
#include<string>
#include<cstring>
#include<string>
#include<algorithm>
#include<vector>
#include<cctype>
#include<map>
#include<set>
#include<queue>
#include<numeric>
#include<iomanip>


using namespace std;

typedef long long ll;
ll a, m, b;
ll phi(ll n) {
	ll ans = n;
	for (ll i = 2; i<= n/i; i++) {
		if (n % i == 0) {
			ans = ans / i * (i - 1);
			while (n % i == 0)n /= i;
		}
	}
	if (n != 1)ans =( ans / n) * (n - 1);
	return ans;
}
inline ll read(ll n) {
	ll f = 0,x=0;
	char c = getchar();
	while (!isdigit(c))c=getchar();
	while (isdigit(c)) {
		x = x * 10 + c-'0';
		if (x >= n)f = 1;
		x %= n;
		c = getchar();
	}
	return x+(f == 1 ?  n : 0);
}
ll fast_pow(ll a,ll b,ll m) {
	ll s = 1;
	while (b) {
		if (b & 1)s = s * a % m;
		a = a * a % m;
		b >>= 1;
	}
	return s;
}
int main() {
	cin >> a >> m;
	ll mns = phi(m);
	b = read(mns);
	cout<<fast_pow(a, b, m);
}

Namespace(命名空间)是一种在编程中用来区分各种不同命名元素的机制。它可以将不同的程序元素(变量、函数、类等)进行分组,使得它们在同一个命名空间下能够互相区分。在不同的命名空间中,可以定义具有相同名称的元素,而不会产生冲突。 快读namespace可以理解为快速了解命名空间的意思。在当前广泛使用的编程语言中,命名空间是一种非常常见的概念,如C++中的命名空间、Python中的模块和包、Java中的包等。通过快速了解命名空间,我们可以更好地进行模块化的开发,避免命名冲突,提升代码的可读性、可维护性和可重用性。 了解命名空间的重要性在于,它可以分隔不同模块之间的代码,使得每个模块可以独立开发和测试。在大型项目中,各个模块的开发往往由不同的开发人员负责,通过使用命名空间,可以避免不同模块中的命名冲突,同时也方便了代码的组织和管理。 在编程中,通过使用命名空间,我们可以更好地控制程序的作用域,避免全局变量的滥用。命名空间可以使得变量和函数的作用范围被限制在一个特定的范围内,有利于代码的结构化和模块化。 总而言之,快读命名空间是指通过了解命名空间的概念和使用方法,可以更好地进行模块化开发,提升代码的可读性和可维护性。同时,对于从事软件开发的人来说,掌握命名空间的概念也是非常重要的。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

louisdlee.

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值