欧拉函数和最大公约数

分析:如果两个数的最大公约数是一个质数p,那么这两个数都除以p,得到的两个数的最大公约数一定是1.

反证法:如果得到的两个数的最大公约数不是1,那么把此时的最大公约数乘以上边的最大公约数,得到的一定比上述的最大公约数大,那么上述的最大公约数就不是最大那两个数的最大公约数,所以结论错误。即得到的两个数的最大公约数一定是1.

由于发现两个数都除以p之后,得到的数的最大公约数是1,那么我们可以想到欧拉函数,此时就可以先处理欧拉函数和欧拉函数的前缀和,然后枚举1~n的所有质数,每次求1~n/p(下取整)中与n/p(下取整)互质的个数,由于(1,2),(2,1)属于两个那么还需要乘以2,(1,1)(1,1)属于1个,最后还得减去1.

cpp 复制代码
#include<bits/stdc++.h>

using namespace std;

const int N = 1e7 + 10;

int hpi[N];
int primes[N],cnt;
bool st[N];
int n;
long long s[N];

void init()
{
    hpi[1]=1;
    
    for(int i=2;i<=n;i++)
    {
        if(!st[i]) 
        {
            primes[cnt++]=i;
            hpi[i]=i-1;
        }
        
        for(int j=0;primes[j]<=n/i;j++)
        {
            st[primes[j]*i]=true;
            if(i%primes[j]==0)
            {
                hpi[primes[j]*i]=primes[j]*hpi[i];
                break;
            }
            hpi[i*primes[j]]=hpi[i]*(primes[j]-1);
        }
    }
    
    for(int i=1;i<=n;i++) s[i]=s[i-1]+hpi[i];
}
int main()
{
    cin>>n;
    
    init();
    long long res=0;
    for(int i=0;i<cnt;i++)
    {
        int p=primes[i];
        res+=(2*s[n/p]-1);
    }
    cout<<res<<endl;
    return 0;
}
相关推荐
房开民6 小时前
可变参数模板
java·开发语言·算法
不知名的忻6 小时前
Morris遍历(力扣第99题)
java·算法·leetcode·morris遍历
状元岐7 小时前
C#反射从入门到精通
java·javascript·算法
_深海凉_7 小时前
LeetCode热题100-除了自身以外数组的乘积
数据结构·算法·leetcode
Kk.08028 小时前
项目《基于Linux下的mybash命令解释器》(一)
前端·javascript·算法
SteveSenna8 小时前
Trossen Arm MuJoCo自定义1:改变目标物体
人工智能·学习·算法·机器人
yong99909 小时前
IHAOAVOA:天鹰优化算法与非洲秃鹫优化算法的混合算法(Matlab实现)
开发语言·算法·matlab
米粒110 小时前
力扣算法刷题 Day 42(股票问题总结)
算法·leetcode·职场和发展
浅念-12 小时前
从LeetCode入门位运算:常见技巧与实战题目全解析
数据结构·数据库·c++·笔记·算法·leetcode·牛客
CoovallyAIHub12 小时前
无人机拍叶片→AI找缺陷:CEA-DETR改进RT-DETR做风电叶片表面缺陷检测,mAP50达89.4%
算法·架构·github