hiho第九十六周 数论五·欧拉函数
来源:互联网 发布:如何利用月经减肥 知乎 编辑:程序博客网 时间:2024/04/27 20:10
描述
小Hi和小Ho有时候会用密码写信来互相联系,他们用了一个很大的数当做密钥。小Hi和小Ho约定了一个区间[L,R],每次小Hi和小Ho会选择其中的一个数作为密钥。
小Hi:小Ho,这次我们选
小Ho:恩,小Hi,这个
小Hi:这个
假设
φ(n) 表示1…n−1 中与n 互质的数的个数。对于[L,R] 中的任意一个除K以外的整数y ,满足φ(K)≤φ(y) 且φ(K)=φ(y) 时,K<y 。
也即是
小Ho:噫,要我自己算么?
小Hi:没错!
小Ho:好吧,让我想一想啊。
<几分钟之后…>
小Ho:啊,不行了。。感觉好难算啊。
小Hi:没有那么难吧,小Ho你是怎么算的?
小Ho:我从枚举每一个L,R的数i,然后利用辗转相除法去计算[1,i]中和i互质的数的个数。但每计算一个数都要花好长的时间。
小Hi:你这样做的话,时间复杂度就很高了。不妨告诉你一个巧妙的算法吧:
欧拉函数
小Hi:刚刚我所描述的
小Ho:又是欧拉么!
小Hi:毕竟是伟大的数学家,所以以他名字命名的东西很多啦。
对于
- 若n为素数,则
φ(n)=n−1
显然,由于
- 若
n=pk,p 为素数(即n 为单个素数的整数幂),则φ(n)=(p−1)∗pk−1
因为
- 若p和q互质,则
φ(p∗q)=φ(p)∗φ(q)
对于所有小于
对于
因为
所以对于每一个固定的
下面我证明一个引理:
假设
假设
由于
则
因此
假设不成立,所以
因此对于任意一个确定的
同理,由
综上,当
满足条件的r一共用
由此得证:
这一段证明不是太好理解,小Ho你一定要自己推导一遍哦。
小Ho:好。
小Hi:在上面这些性质的基础上我们能到推导出两条定理:
- 若p为质数,n为任意整数:若
p 为n 的约数,则φ(n∗p)=φ(n)∗p ;若p 为不为n 的约数,则φ(n∗p)=φ(n)∗(p−1)
若p 为n 的约数,且p 为质数。则我们可以将n 表示为pk∗m 。m 表示其他和p 不同的质数的乘积。
显然有
由
根据这两条定理,当我们得到一个
小Ho:嗯…我想起了,这不是我们使用欧拉筛法时一样的算法么?
小Hi:没错!因此我们只需要在欧拉筛代码的基础上做一个小改动,就可以得到递推求解
isPrime[] = trueprimeList = []phi = [] // phi[n]表示n的欧拉函数primeCount = 0For i = 2 .. N If isPrime[i] Then primeCount = primeCount + 1 primeList[ primeCount ] = i phi[i] = i - 1 // 质数的欧拉函数为p-1 End If For j = 1 .. primeCount If (i * primeList[j] > N) Then Break End If isPrime[ i * primeList[j] ] = false If (i % primeList[j] == 0) Then // primeList[j]是i的约数,φ(n*p) = φ(n) * p phi[ i * primeList[j] ] = phi[i] * primeList[j]; Break Else // primeList[j]不是i的约数,φ(n*p) = φ(n) * (p-1) phi[ i * primeList[j] ] = phi[i] * (primeList[j] - 1); End If End IfEnd For
小Ho:因为欧拉筛的时间复杂度是O(n)的,因此求出一个大区间内所有数的欧拉函数也只用了O(n)的时间。接下来再使用O(n)的枚举就可以求得最小的K了。我知道该怎么做了!
代码:
#include<iostream>using namespace std;const int max_n = 5000000;int isPrime[max_n+1];int primeList[max_n+1];int phi[max_n];void Euler_function(){ for (int i = 0; i <= max_n; i++) isPrime[i] = true, phi[i] = 1; for (int i = 2; i <= max_n; i++) { if (isPrime[i]) { primeList[++primeCount] = i; phi[i] = i - 1; } for (int j = 1; j <= primeCount; j++) { if (i*primeList[j] > max_n) break; isPrime[i*primeList[j]] = false; if (i%primeList[j] == 0) { phi[i*primeList[j]] = phi[i] * primeList[j]; break; } else phi[i*primeList[j]] = phi[i] * (primeList[j] - 1); } }}int main(){ int L, R;cin >> L >> R; int min = L; Euler_function(); for (int i = L+1; i <= R; i++) { if (phi[i] < phi[min]) min = i; } cout << min<< endl; return 0;}
- hiho第九十六周 数论五·欧拉函数
- hihocoder第九十六周 数论五·欧拉函数
- HIHO #1298 : 数论五·欧拉函数
- hiho 1298 数论·五 欧拉函数 (欧拉函数筛选板子)
- hiho第九十六周
- hihoCoder 数论五·欧拉函数
- hiho第九十七周 数论六·模线性方程组
- hiho第九十五周 数论四·扩展欧几里德
- hihocoder 1298 : 数论五·欧拉函数
- 数论欧拉函数
- 欧拉函数(数论)
- 数论-欧拉函数
- 欧拉函数---数论
- hiho一下 第九十二周 数论一·Miller-Rabin质数测试
- hiho一下 第九十二周 数论一·Miller-Rabin质数测试
- hiho一下 第九十三周 数论二·Eular质数筛法
- hiho一下 第九十三周 数论二·Eular质数筛法
- hiho第九十三周 数论二·Eular质数筛法
- VO ,PO ,BO,QO, DAO ,POJO,概念
- 时间下拉框——时间控件四部曲(3)
- js中的$和#
- la 5739
- 有一副由NxN矩阵表示的图像,这里每个像素用一个int表示,请编写一个算法,在不占用额外内存空间的情况下(即不使用缓存矩阵),将图像顺时针旋转90度。
- hiho第九十六周 数论五·欧拉函数
- Servlet初识
- 请编写一个算法,若MxN矩阵中某个元素为0,则将其所在的行与列清零。 给定一个MxN的int[][]矩阵(C++中为vector>)mat和矩阵的阶数n,请返回完成操作后的int[][]矩阵(C++中
- spring boot 集成springmvc视图技术
- HdU 2544 最短路
- 假定我们都知道非常高效的算法来检查一个单词是否为其他字符串的子串。请将这个算法编写成一个函数,给定两个字符串s1和s2,请编写代码检查s2是否为s1旋转而成,要求只能调用一次检查子串的函数。
- 73. Set Matrix Zeroes
- TCP协议详解
- Linux.fedora23.WORKSTATION简易安装教程(WIN10+Linux双系统)