UVA 1635 Irrelevant Elements(杨辉三角+递推式求组合数+算数基本定理)

题目大意:

       1.输入n 和 m, n个数为任意的a1, a2, a3, ...., an,利用杨辉三角的求算方式将这n个数不断加和,求到最后只有一个数

       2.然后用最后的和对m 取余数,问哪一个数与m的余数没有关系

解题思路:

      1.把题目看懂的大概思路就是求出这n个数的杨辉三角的系数,然后看这里面的哪一个数,是m的因子,(因为如果是的话那么它乘上ai也一定还能整除m一定对m的余数没有作用),因为n 能到10000,所以组合数也会非常的大,不能用来%m,只能用算数基本定理,用每一项的素因子及其指数来判断


递推法求组合数C(n,k):

int c(int n, int k)
{
    int result = 1;
    for(int i = 0; i < k; i++)
        result = result * (n-i) / (i+1);
    return result;
}

Code;

#include
#include
#include
#include
using namespace std;
int fac[105][2];
int m, n;
int num;

//分解质因子。算数分解定理
void factory()
{
    for(int i = 2; i*i <= m; i++)
    {
        if(m % i == 0)
        {
            fac[++num][0] = i;
            fac[num][1] = 0;
            do
            {
                fac[num][1]++;
                m /= i;
            }while(m%i == 0);
        }
    }
    if(m > 1) //分解到最后还有剩余,本身就是一个素数
    {
        fac[++num][0] = m;
        fac[num][1] = 1;
    }
}

int c[105];
bool check(int n, int i)
{
    int x = n-i; // (n-1 - i +1)
    int y = i;

    for(int i = 1; i <= num; i++)
    {
        int p = fac[i][0];
        while(x % p == 0)
        {
            x /= p;
            c[i]++;
        }
        while(y % p == 0)
        {
            y /= p;
            c[i]--;
        }
    }
    for(int i = 1; i <= num; i++)
        if(c[i] < fac[i][1])
        return false;
    return true;
}
int A[100005];
int main()
{
    while(scanf("%d %d", &n, &m) != EOF)
    {
        num = 0;
        factory();
        memset(c, 0, sizeof(c));
        int ii = 0;
        for(int i = 1; i < n-1; i++)
        {
            if(check(n,i))
                A[ii++] = i+1;
        }
        cout << ii << endl;
        for(int i = 0; i < ii; i++)
        {
            if(i == 0)
                cout << A[i];
            else
                cout << " " << A[i];
        }
        cout << endl;
    }
    return 0;
}


你可能感兴趣的:(数学)