生成元(Digit Generator)
题目:如果x加上y的各个数字之和得到y,就说x是y的生成元。给出n(1<=n<=100000),求最小生成元。无解输出0。例如,n=216,121,2005时的解分别为198,0,1979.
一、枚举所有的m<n,找到数n的生成元
int ans[maxn];
int main()
{
int n,m,T;
scanf("%d",&T);
while(T--){
scanf("%d",&n);
for( m=1;m<=n;m++)
{
if(n<10){
if(2*m==n){
printf("%d的生成元是%d\n",n,m);
break;
}
}
else if(n<100){
if((m+m/10+m%10)==n){
printf("%d的生成元是%d\n",n,m);
break;}
}
else if(n<1000){
if((m+m/100+(m/10)%10+m%10)==n){
printf("%d的生成元是%d\n",n,m);
break;}
}
else if(n<10000){
if((m+m/1000+((m/100)%10)+((m/10)%10)+m%10)==n){
printf("%d的生成元是%d\n",n,m);
break;}
}
else {
if((m+m/10000+((m/1000)%10)+((m/100)%10)+((m/10)%10)+m%10)==n){
printf("%d的生成元是%d\n",n,m);
break;}
}
}
if(m==n+1) printf("0");
}
return 0;
}
此方法较为繁琐,且效率不够高。
二、一次性枚举100000以内的所有正整数m,标记“m加上m的各个数字之和得到的数有一个生成元时m”,最后查表即可。
#include<stdio.h>
#include<string.h>
#define maxn 100005
int ans[maxn];
int main()
{
int T,n;
memset(ans,0,sizeof(ans));
for(int m=1;m<maxn;m++){
int x=m,y=m;
while(x>0){
y+=x%10;
x=x/10;
}
if(ans[y]==0 || m<ans[y] ) ans[y]=m;
}
scanf("%d",&T);
while(T--){
scanf("%d",&n);
printf("%d\n\n",ans[n]);
}
return 0;
}