题目大意:
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<iostream>
#include<cstdio>
#include<cstring>
#include<cstdlib>
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;
}