初等数论_6 2016.4.15

本文介绍了欧拉函数的基本概念、性质,并通过HDU的三道竞赛题目展示了欧拉函数在解决数论问题中的应用,如计算互质数量、最大公约数等。

四、欧拉函数

1、定义
在数论中,对于正整数n,欧拉函数就是小于n的数中与n互质的数的数目。
此函数以其首名研究者欧拉命名(Ruler’so totient function),它又称为Euler’s totient function、φ函数、欧拉商数等。

例如φ(8)=4,因为1,3,5,7均和8互质。
φ(24)=8,因为1, 5, 7, 11, 13, 17, 19, 23均和 24 互质。

通式:
这里写图片描述
其中p1, p2……pn为x的所有质因数,x是不为0的整数。
φ(1)=1(唯一和1互质的数就是1)。

注意:
每种质因数只一个。
比如12=2*2*3那么φ(12)=12 * (1-1/2)* (1-1/3)=4

2、基本性质
①若N是质数p的k次幂,φ(N)=p^k-p^(k-1)=(p-1)p^(k-1),因为除了p的倍数外,其他数都跟N互质
②当N是质数时,φ(N) = N-1
③除了N=2,φ(N)都是偶数
④小于N且与N互质的所有数的和是φ(n)*n/2
⑤欧拉函数是积性函数——若m,n互质,φ(m*n)=φ(m)*φ(n)
⑥当N为奇数时,φ(2*N)=φ(N)

HDU 2824 The Euler function

解题思路:
筛选法打欧拉函数表

#include <iostream>
#include <cstdio>

using namespace std;

typedef long long LL;

const int maxn = 3000000 + 10;
int Euler[maxn];

void Init(void);

int main()
{
    Init();
    int a, b;

    while (scanf("%d%d", &a, &b) != EOF) {
        LL ans = 0;
        for (int i=a; i<=b; ++i) {
            ans += Euler[i];
        }
        printf("%I64d\n", ans);
    }
    return 0;
}

void Init(void)
{
    for (int i=1; i<maxn; ++i) {
        Euler[i] = i;
    }
    for (int i=2; i<maxn; ++i) {
        if (Euler[i] == i) {
            for (int j=i; j<maxn; j+=i) {
                Euler[j] = Euler[j] / i * (i-1);
            }
        }
    }
}

HDU 2588 GCD

题意:
Given integers N and M, how many integer X satisfies 1<=X<=N and Gcd(X,N)>=M.

解题思路:
设ai为N的大于等于M的约数
那么答案就是n/ai的欧拉函数之和

#include <iostream>
#include <cstdio>

using namespace std;

int Euler(int n);

int main()
{
//    freopen("in.txt", "r", stdin);
    int T;

    scanf("%d", &T);
    int N, M;
    while (T--) {
        scanf("%d%d", &N, &M);
        int ans = 0;
        for (int i=1; i*i<=N; ++i) {
            if (N%i == 0) {
                if (i >= M) {
                    ans += Euler(N/i);
                }
                if (N/i != i && N/i >= M) {
                    ans += Euler(i);
                }
            }
        }
        printf("%d\n", ans);
    }
    return 0;
}

int Euler(int n)
{
    int ret = n;
    for (int i=2; i*i<=n; ++i) {
        if (n%i == 0) {
            ret = ret/i*(i-1);
            while (n%i == 0) {
                n /= i;
            }
        }
    }
    if (n != 1) {
        ret = ret/n*(n-1);
    }
    return ret;
}

HDU 4983 Goffi and GCD

转自http://blog.youkuaiyun.com/yanghuaqings/article/details/47167875

题意:
给出一组n和k,求解满足公式:gcd(n-a,n)*gcd(n-b,n)=n^k的(a,b)的个数
结果对(1e9+7)取模

解题思路:

Gcd(x, n) = Gcd(n-x, n)

Gcd(n-a,n)*Gcd(n-b,n)=n^k可以化为Gcd(a,n)*Gcd(b,n)=n^k

gcd(a,n)<=n,gcd(b,n)<=n
所以gcd(a,n)*gcd(b,n)<=n^2

当n=1时,只有1解
当k>2时,无解
当k=2时,只有a=b=n时,Gcd(a,n)=n,Gcd(b,n)=n,Gcd(a,n)*gcd(b,n)=n^2,即只有1解

当k=1时,就是求Gcd(a,n)*Gcd(b,n)=n

如果Gcd(a,n)=x,则Gcd(b,n)=n/x,这里只要枚举x,求n/x即可
x是a和n的最大公约数,那么x就是n的因数,因此枚举n的因数就可以了

对于每一个x可能会有多个a(1<=a<=n)存在,使得Gcd(a,n)=x,假设存在ma个
那么对于每一个n/x,同样会有多个b(1<=b<=n)存在,使得Gcd(b,n)=n/x,假设存在mb个
ma,mb可以用欧拉函数求解

那么对于一个x,如果x*x!=n,那么就存在2*ma*mb
对结果
如果x*x==n,那么就存在ma*mb对结果。

#include <iostream>
#include <cstdio>

using namespace std;

typedef long long LL;

int mod = 1e9 + 7;

LL Euler(int n);

int main()
{
//    freopen("in.txt", "r", stdin);
    int n, k;

    while (scanf("%d%d", &n, &k) != EOF) {
        if (n == 1) {
            printf("1\n");
        } else if (k > 2) {
            printf("0\n");
        } else if (k == 2) {
            printf("1\n");
        } else {
            LL ans = 0;
            for (int i=1; i*i<=n; ++i) {
                if (n%i == 0) {
                    if (i*i == n) {
                        ans += Euler(n/i) * Euler(i);
                    } else {
                        ans += 2 * Euler(n/i) * Euler(i);
                    }
                    ans %= mod;
                }
            }
            cout<<ans<<endl;
        }
    }
    return 0;
}

LL Euler(int n)
{
    LL ret = n;
    for (int i=2; i*i<=n; ++i) {
        if (n%i == 0) {
            ret = ret/i*(i-1);
            while (n%i == 0) {
                n /= i;
            }
        }
    }
    if (n != 1) {
        ret = ret/n*(n-1);
    }
    return ret%mod;
}
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值