求约数

试除法求约数:

#include <iostream>
#include <algorithm>
#include <vector>
using namespace std;
int n;
vector<int> get_divisors(int x)
{
    vector<int> res;
    for(int i = 1;i <= x / i; i++)
    {
        if(x % i == 0)
        {
            res.push_back(i);
            if(i != x / i) res.push_back(x / i);
        }
    }
    sort(res.begin(), res.end());
    
    return res;
}
int main()
{
    cin>>n;
    while(n--)
    {
        int x;
        cin>>x;
        vector<int> res = get_divisors(x);
        for(auto i : res)
            cout<<i<<" ";
        cout<<endl;
    }
}

从1到sqrt(x)来不停的判断是否整除,是的话那么就加进来,加一个特判:i != x / i

约数个数:

//约数个数、约数之和 算数整除定理;N = p1^a1 + p2^a2 + ... + pn^an
#include <iostream>
#include <unordered_map>
using namespace std;
const int mod = 1e9 + 7;
typedef unsigned long long ULL;
int main()
{
    int n;
    cin>>n;
    unordered_map<int, int> primes;
    while(n--)
    {
        int x;
        cin>>x;
        for(int i = 2; i <= x / i; i++)
        {
            while(x % i == 0)
            {
                x /= i;
                primes[i] ++;
            }
        }
        if(x > 1) primes[x]++;
    }
    ULL res = 1;
    for(auto i : primes)
    {
        res = res * (i.second + 1) % mod;
    }
    cout<<res<<endl;
}
N = p1^a1 + p2^a2 + ... + pn^an

 

约数个数等于各个约数的指数+1相乘,(a1+1)*(a2+1)……(an+1)

因为每个数的取值范围都是0~a1、0~a2、0~an 根据组合定律 所以约数的个数就是(a1 + 1) * (a2 + 1)  * (an + 1)。

p约数之和等于(p1^0+p1^1+……p1^a1)……(pk^0+pk^1+……+pk^ak),将之展开实际上就是每个约数然后相加。

以样例为例子:

3
2
6
8
约数分别为:1、2、3、4、6、8、12、16、24、32、48、96
2 * 6 * 8 = 2^5 * 3 约数的个数就是(5 + 1) * (1 + 1) = 12;
约数之和就是:(2^0 + 2^1 + 2^2 + 2^3 + 2^4 + 2^5) * (3^0 + 3^1) = 63 * 4 = 252;
(2^0 + 2^1 + 2^2 + 2^3 + 2^4 + 2^5) * (3^0 + 3^1) = 2^0 * 3^0 + 2^0 * 3^1 + 2^1 * 3^0 + 2^1 * 3^1 + 2^2 * 3^0 + 2^2 * 3^1……2^5 * 3^0 + 2^5 * 3^1
实际上展开的就是每个约数然后加起来。
 约数之和代码:
#include <iostream>
#include <unordered_map>
using namespace std;
const int mod = 1e9 + 7;
typedef unsigned long long ULL;
int main()
{
    int n;
    cin>>n;
    unordered_map<int, int> primes;
    while(n--)
    {
        int x;
        cin>>x;
        for(int i = 2; i <= x / i; i++)
        {
            while(x % i == 0)
            {
                x /= i;
                primes[i] ++;
            }
        }
        if(x > 1) primes[x]++;
    }
    ULL res = 1;
    for(auto i : primes)
    {
        int p = i.first, a = i.second;
        ULL t = 1;
        while(a--) t = (t * p + 1) % mod;
        res = (res * t) % mod;
    }
    cout<<res<<endl;
}

 

t = (t * p + 1) % mod;这是秦九韶算法

求最大公约数:

#include <iostream>
using namespace std;
int gcd(int a, int b)
{
    return b == 0 ? a : gcd(b,a%b);
}
int main()
{
    int n;
    cin>>n;
    while(n--)
    {
        int a,b;
        cin>>a>>b;
        cout<<gcd(a,b)<<endl;
    }
}

因为有这样的性质:

gcd(a, b) = gcd(b, a%b) 左右两边的公约数相等,如有三个整数d, a, b:

1)首先从左边推到右边:d | a, d | b,那么d | ax + by。a % b = a - a / b * b; (a / b = c向下取整)a % b = a - c * b(x = 1, y = -c)所以 d | a % b;

2)再从右边推到左边 :d | b, d | (a % b) = d | ( a - c * b), 所以 d | ((a - c * b) + c * b) = d | a,得证。(这里a-c*b=a,x=1,b=c*b,y=1)

求最大公约数一句话:return b == 0 ? a : gcd(b, a%b) ; 或者 return b ? gcd(b, a % b) : a;

posted @ 2020-04-17 19:57  龙雪可可  阅读(265)  评论(0编辑  收藏  举报
****************************************** 页脚Html代码 ******************************************