水仙花数问题:
有个比较经典的问题,就是水仙花数问题,简单来说就一个三位数,它的各位数字的立方和等于其本身,比如153=1^3+5^3+3^3
Case1:现在要求输出所有在m和n范围内的水仙花数
算法分析:
首先水仙花必须是三位数,那么为数据的范围自然而然是【100,999】,可知每个数都是三位数,那只需要令a,b,c分别等于三个数,然后求出它们的立方和是否等于这个三位数,如果是则就是水仙花数。那只需要枚举这个区间内所有的数,统计水仙花数的个数即可。
算法如下:
main(){
int m,n,I,a,b,c,d=0,k;
while(cin>>m>>n){
if(n<m){
i=n;
n=m;
m=i;
}
}
for(i=m,d=0;i<=n;i++){
a=i%10;//取出个位
b=(i/10)%10;//取出十位
c=i/100;//取出百位
if(a*a*a+b*b*b+c*c*c=i){
d++;
}
for(i=m,k=0;i<=n;i++){
a=i%10;//取出个位
b=(i/10)%10;//取出十位
c=i/100;//取出百位
}
if(a*a*a+b*b*b+c*c*c=i){
cout<<i;k++;
if(k<d){
cout<<"";
}
}
if(d==0){
cout<<"no"<<endl;
}
}
}
复制代码
完数问题
完数的定义:如果一个大于1的正整数的所有因子之和等于它的本身,则称这个是完数,比如6 =1+2+3 既是6的因子,这些因子之和也是等于6,再比如28=1+2+4+7+14
Case2:判断两个数正整数之间完数的个数
算法分析:
根据定于预处理的枚举出1-10000以内的完数有哪些,然后用各数ans【i】存1~i以内的完数个数,这样就可以直接判断某个区间内的完数个数。 比如区间【a,b】那答案就是ans【b】-ans【a-1】
算法设计:
#define mm 1009
int ans[mm]
bool is_ok(int m){
int sum=0;
int ms=(int) sqrt((float)m)+1;//o枚举m的除1以外的每个因子
for(int i=2;i<ms;i++){
if(m%i==0){
sum+=i+m/i;
}
sum++;//如果所有因子的和等于这个数m,就说明m为完数
if(sum==m){
return true;
}eles{
return false;
}
}
}
main(){
ans[1]=1;//ans[i] 表示1~i有多少个完数
for(int i=1;i<10000;i++){
if(is_ok(i))
ans[i]=ans[i-1]+1;
else ans[i]=ans[i-1];
int cas,a,b;
cin>cas;
while(cas--){
cin>>a>>b;
if(a>b){
int z;
z=b;
a=b;
b=z;
}
cout<<ans[b]-ans[a-1];
}
}
复制代码
其实这类问题更多是数学思维上的考察,算法设计并不难,但是如何将数学问题转化成算法设计的核心,如何找到对运行效率要求较高的大规模的数据处理问题时,必须要多动动脑筋,因此冰冻三尺非一日之寒。