最 小 公 倍 数 算 法 分 析

  1. 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
  2. 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
  3. 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。

算法设计与分析:第一章算法分析介绍 1.1求任意三个已知数的最小公倍数

求任意三个已知数的最小公倍数:

自己分析:

分治,看看先求出两个数的最小公倍数,例如8和12最小公倍数为24,再求24与28最小公倍数:

这是可行的。

另一种方法:

12= 2^2 * 3

28 = 2^2 * 7

最小公倍数 = 2^3 * 3 * 7 = 8 * 21 = 168

所以应该先分解质因数,然后用一个数组保存质因子的指数,分解的时候覆盖分解;

如果当前质因子的指数比原来存储的指数要大,则更新

n个数,每个数分解O(n),

12 = 2^2 * 3

最大公约数 = 2^2找到最大的公共指数部分

可以用素数筛选法,选择出10000个素数,来做

iPrimeArr[iNum]中存放的是{2,3,5,7}这种

大牛分析:

为了避免因数重复计算,每次都需要除掉3个整数中已经找到的因数

(即用因数法)

去除含有它的整数。因此需要记录i具体是哪个数的个数,要对哪个数进行整数。

例如:2是2,5,6中2,6两个数的因数,因此要用2,6去除以2得到新的一组数

例如:8 12 28,

1)找到8与12的公因数4,除掉得2,3,28;记录4

2)找2与28的公因数2,除掉得1,14,3;记录2

3)找到3与14的公因数1,除掉得3,14,1,记录1

判断3个公因数是否存在包含,关系,

最后用剩余的三个数的乘积乘以3次公因数=1*3*14*4*2*1=42*8=336 #include stdio.h

#include string.h

#include math.h

const int MAXSIZE = 10000;

int isPrime(int* pArr,int iLen,int* pPrimeArr)

--memset(pArr,0,sizeof(pArr));--默认全部是不是素数【用0表示】,然后将是合数的翻过来

int m = sqrt(0.5 + iLen);

int iCnt = 0 ;

for(int i = 2 ; i = iLen ; i++)--素数筛选法

if(pArr[i] == 0)--如果当前数是素数,置其倍数为合数;如果当前数

是合数,则不需要重新设置

pPrimeArr[iCnt++] = i;--存储素数

for(int k = i*i ; k = iLen ; k += i)--从平方开始而不从2i开始的原因是,避免重复运算,2*2=4,

pArr[k] = 1;--将素数的倍数置为合数的标记

return iCnt;

int divide(int iNum,int* pTimeArr,int* pPrimeArr,int iPrimeLen) for(int i = 0 ;i iPrimeLen ; i++)

int iCnt = 0;

while( iNum % pPrimeArr[i] == 0)

iNum -= pPrimeArr[i];

if(iCnt pTimeArr[i])

pTimeArr[j] = iCnt;--相同的质因子保留最大的,记录质因数的个数return j;

long long power(int iNum,int iIndex)

if(iIndex == 1)--递归出口

return iNum;

if(iIndex == 0)

return 1;

long long iRet = power(iNum,iIndex-2);

iRet *= iRet;

if(iIndex % 2 == 1)

iRet *= iNum;

return iRet;

long long result(int* pTimeArr,int* pPrimeArr,int iLen) long long lRet = 1;

for(int i = 0 ; i iLen ; i++)

lRet *= power(pPrimeArr[i],pTimeArr[i]);

return lRet;

int max(int a,int b)

return a b ? a : b;

int gcd(int maxNum,int minNum)

return minNum == 0 ? maxNum : gcd(minNum,maxNum%minNum); void swap(int* pNum1,int* pNum2)

int iTemp = *pNum1;

*pNum1 = *pNum2;

*pNum2 = iTemp;

int result2(int a,int b,int c)

swap(a,b);

int lcm = a*b-gcd(a,b);--求取a,b最小公倍数

if(lcm c)

swap(lcm,c);

return lcm*c-gcd(lcm,c);

void process()

int a,b,c;

int iTimeArr[MAXSIZE];

int iNumArr[MAXSIZE];

int iLen = 0;

int iArr[MAXSIZE];

int iPrimeArr[MAXSIZE];

memset(iArr,0,sizeof(iArr));

int iPrimeLen = isPrime(iArr,MAXSIZE,iPrimeArr);

while(EOF != scanf("%d %d %d",a,b,c))

memset(iTimeArr,0,sizeof(iTimeArr));--分解的质因子范围为当前数分解到1为止

iLen = max(iLen, divide(a,iTimeArr,iPrimeArr,iPrimeLen));

iLen = max(iLen, divide(b,iTimeArr,iPrimeArr,iPrimeLen));

iLen = max(iLen, divide(c,iTimeArr,iPrimeArr,iPrimeLen));

printf("%lld",result(iTimeArr,iPrimeArr,iLen));

printf("%d",result2(a,b,c));

int main(int argc,char* argv[])

process();

getchar();

return 0;

lcm = min * (min + 1) - fun_gcd(min + 1, min);

例如:2是2,5,6中2,6两个数的因数,因此要用2,6去除以2得到新

相关文档
最新文档