一、完成的概念
什么是完数?如果一个数恰好等于其因子之和,这个数就称为完数。“完数”这个名称具有神秘的色彩,意思是“完美的数”。如古代意大利人就把6看做属于爱神维纳斯的数,它象征着美满的婚姻。
例如,数6,其等于1+2+3,而且,1、2、3又都是6的整约数,也就是说,这些数都能除尽6。因此,数6就是一个完数。这是第1个完数,而第2个完数是28,因为:1+2+4+7+14=28.
完数在自然数中很少。据统计,在1到4000万这么多的自然数里,只有7个完数,依次是6、28、496、8128、120 816、2 096 128、33 550 336。
根据以上分析,如果要证明以上7个数是完数,其演算过程相当繁琐,并且计算过程中容易出错,而如果使用计算机来进行演算,却可以达到既快又准确的效果。
题目:现要求编写程序,找出10 000以内的所有完数。
解题思路:要找出10 000以内的所有完数,需将这些数逐个进行检查,具体的检查过程就是(假设需要检查的数为n):
(1)用n去除以1~n的所有整数,将能整除的被除数保存到一个数组中,作为n的一个因子。
(2)用数n减去该因子,以方便计算各因子之和是否正好等于n。
(3)继续重复步骤1和步骤2,直至将所有整数除完为止。
(4)判断各因子之和是否等于数n,若相等,则数n为完数,输出该数和各因子。
二、程序实现
#include <stdio.h>
int PerfectNum(long n)
{
if (n <= 0) {
printf("数字范围不能小于0!\n");
return 0;
}
long p[300];
long i, num, count, s, c=0;
for (num=2; num<=n; num++) {
count = 0;
s = num;
for (i=1; i<num; i++) {
if (num%i == 0) {
p[count++] = i;
s -= i;
}
}
if (s == 0) {
printf("%4ld是一个完数,因子是:", num);
printf("%ld=%ld", num, p[0]);
for (i=1; i<count; i++)
printf("+%ld", p[i]);
printf("\n");
c++;
}
}
printf("共找到%ld个完数\n", c);
return 1;
}
int main(int argc, char *argv[])
{
long n = 10000;
PerfectNum(n);
return 0;
}
运行结果:
对以上程序,还可进一步优化,以提高程序的执行速度。第11~18行为一个循环,该循环主要的目的是分解数num的因子,而num在第10行中保存在变量s中,第16行需要减去分解的因子,如果s的值已经小于或等于0,则不需要进行后续的分解。
例如,对于数8,若按第11~18行的循环程序,将分别用数8去除以1、2、3、4、5、6、7,再判断余数是否为0,若为0表示能整除,就保存一个因子,然后在变量s中减去该因子。
从图6-1所示的结果可看出,每一个完数的最后一个因子就是该数除以2的商,因此,循环终值可以设置为所求完数的一半,这样,可减少一半的循环量。可将第14行的程序改为以下形式
for (num=2; num<=n/2+1; num++)