2.1.1算法案例分析(1)算法的基本思想例1在电视台的某个娱乐节目中,要求参与者快速猜出物品价格.主持人出某件物品,参与者每次估算出一个价格,主持人只能回答高了、低了或者正确.在某次节目中,主持人出示了一台价值在1000元以内的随身听,并开始了竞猜.下面是主持人和参与者之间的一段对话:参与者:800元!主持人:高了!参与者:400元!主持人:低了!参与者:600元!主持人:低了!……如果你是参与者,你接下来会怎么猜?分析理解•如果用P表示商品的价格,则参与者的猜测结果为:•由主持人的第一次回答,P在0~800元之间;•由主持人的第二次回答,P在400~800元之间;•由主持人的第三次回答,P在600~800之间.根据参与者的猜测,我们知道,参与者首先需要确定的是商品价格的范围,数学上一般可以用区间来表示,然后再根据主持人的回答,报出区间中点,将价格区间缩小一半.因此,下一步参与者要猜的数应是700元,然后根据主持人的回答继续猜,直至猜出正确价格.分析理解抽象概括•实际上,可以把上述过程概括如下:1.报出首次价格T1;2.根据主持人的回答确定价格区间:(1)若报价小于商品价格,则商品的价格区间为(T1,1000);(2)若报价大于商品价格,则商品的价格区间为(0,T1);(3)若报价等于商品价格,则游戏结束.3.如果游没有结束,则报出上面确定的价格区间的中点T2.按照上述方法,继续判断,直到游戏结束.像这样的一系列步骤通常称为这个问题的一个算法.分析理解•1.查表判断936是否是素数:(1)如果936是素数,则分解结束;(2)如果不是素数,则进行第2步.2.确定936的最小素因数:2.936=2×4683.查表判断468是否则是素数:(1)如果468是素数,则分解结束;(2)如果468不是素数,则重复上述步骤,确定468的最小素因数.重复进行上述步骤,直到找出936的所有素因数.例2在给定素数表的条件下,设计算法,将936分解成素因数的乘积(4000以内的素数见附录1)解算法步骤如下:1.判断936是否为素数:否.2.确定936的最小素因数:2.936=2×468.3.判断468是否为素数:否.4.确定468最小素因数:2.936=2×2×234.5.判断234是否为素数:否.6.确定234最小素因数:2.936=2×2×2×1177.判断117是否为素数:否.分析理解8.确定117最小素因数:3.936=2×2×2×3×39.9.判断39是否为素数:否.10.确定39的最小素因数:3.936=2×2×2×3×3×1311.判断13是否为素数:13是素数.所以分解结束.分解结果是:936=2×2×2×3×3×13分析理解分析理解按照以上程序,完成了对936的素因数分解.实际上,在给定素数表的基础上,对任意自然数n,都可以按照上述办法进行分解.以上步骤是解决素因数分解问题的一个过程,只要依照这一系列步骤,都能解决这个问题.我们把这一系列步骤成为解决这个问题的一个算法.我们已经学习了对自然数进行素因数分解的方法,下面的算法就是在此基础上设计的.解答这个问题需要按以下思路进行.首先,对两个数分别进行素因数分解:840=23×3×5×7,1764=22×32×72其次,确定两数的公共素因数:2,3,7.接着,确定公共素因数的指数:对于公共素因数2,22是1764的因数,23是840的因数,因此22是这两个数的公因数,这样就确定了公共素因数2的指数为2.同样,可以确定出公因数3和7的指数均为1.这样,就确定了840与1746的最大公因数为:22×31×71=84.例3设计一个算法,求840与1764的最大公因数.例题讲解解算法步骤如下:1.先将840进行素因数分解:840=23×3×5×7;2.然后将1764进行素因数分解:1746=22×32×72;3.确定它们的公共素因数:2,3,7;4.确定公共素因数的指数:公共素因数2,3,7的指数分别为2,1,1;5.最大公因数为:22×31×71=84.例题讲解分析理解以上步骤就是求两个正整数的最大公因数的一个算法.这个算法的思想具有一般性,它可以帮助设计求三个或者三个以上正整数的最大公因数的算法.在这个算法的设计中,我们首先分别对840和1764进行素因数分解,那么,如何将840或1764的所有素因数都找出来呢?例2给出了具体的算法.我们通常会把“对自然数进行素因数分解”的算法,做成一个“程序包”,又称为一个“平台”,在需要“对自然数进行素因数分解”时,把做好的“程序包”拿来使用即可.同样的,我们也可以把“求两个自然数的最大公因数”做成一个“程序包”或“平台”,在解决其他问题时,如果需要“求两个自然数的最大公因数”,我们就可以把做好的程序包直接拿来使用,通常把这种思想称为“平台思想”,“平台”的思想在算法设计中是一个最基本的思想,也是数学中思考问题的一个重要思想.通过以上的例子可以看出,算法是解决某类问题的一系列步骤或程序,只要按照这些步骤执行,都能使问题得到解决.一般来说,“用算法解决问题”都是可以利用计算机帮助完成的.分析理解课堂练习1.模仿例3,设计一个算法,求324,440,556的最大公因数.1.解(1)分别将324,440,556进行素因数分解:324=22×34,440=23×5×11,556=22×139.(2)确定三个数的公共素因数:2(3)确定公共素因数的指数:2(4)最大公因数为:22=4.2.解(1)首先将1356和2400进行素因数分解:1356=22×3×113,2400=25×3×52.(2)确定最小公倍数的素因数:2,3,5,113.(3)确定素因数的指数分别为:5,1,2,1.(4)最小公倍数为:25×3×52×113=271200.课堂练习2.设计算法,求1356和2400的最小公倍数.2.1.1算法案例分析(2)例题解析•例1“韩信点兵”问题韩信是汉高祖刘邦手上的大将,他英勇善战,智谋超群,为建立汉朝立下了汗马功劳.据说他在点兵的时候,为了保住军事机密,不让敌人知道自己部队的实力,采用下述点兵方式:先令士兵从1~3报数,结果最后一个士兵报2;再令士兵从1~5报数,结果最后一个士兵报3;又令士兵从1~7报数,结果最后一个士兵报4.分析理解士兵从1~3报数,最后一个士兵报2.这句话说明士兵的总人数除以3余2.算法的第一步是将所有除以3余2的正整数找出来,按照从小到大的顺序排成一列数.士兵从1~5报数,最后一个士兵报3.这句话说明士兵的总人数除以5余3.算法的第二步是从上列数中找出除以5余3的一列数.最后,在满足前两个条件的上列数中再找出除以7余4的一列数,这列数中最小的数,即为我们所求的数可.这样,韩信很快计算出了自己部队士兵的总人数.请设计一个算法,求出士兵至少有多少人.解算法步骤如下:1.首先确定最小的满足除以3余2的正整数:2;2.依次加3就得到所有除以3余2的正整数:2,5,8,11,14,17,20,23,26,29,32,35,38,41,44,47,50,53,56,…3.在上列数中确定最小的满足除以5余3的正整数:8;分析理解4.然后依次加上15,得到8,23,38,53,…不难看出,这些数既满足除以3余2,又满足除以5余3;5.在第4步得到得以列数中找出满足除以7余4的最小数53,这就是我们要求的数.在完成上述步骤后就找到了所求的数53,这5个步骤称为解决“韩信点兵”问题的一个算法.想一想大家可以想一想,能否在这个算法的基础会上作一些改进,以更快地获得结果?实际上,我们颠倒一下3,5,7的顺序,就可以得到另一个算法;1.首先确定最小的除以7余4的正整数:4;2.依次加7就得到所有除以7余4的正整数;4,11,18,25,32,39,46,53,60,…3.在第2步得到的一列数中确定最小的除以5余3的正整数:18;4.然后依次加上35,得到18,53,88,…5.在第4步得到的一列数中找出最小的满足除以3余2的正整数:53.例题解析例2一位商人有9枚银元,其中有一枚略轻的是假银元.你能用天平(不用砝码)将假银元找出来吗?最容易想到的解决这个问题的一种方法是:把9枚银元按顺序排成一列,先称前2枚,若不平衡,则可找出假银元;若平衡,则2枚银元都是真的,再依次与剩下的银元比较,就能找出假银元.分析理解解按照下列步骤,就能将假银元找出来:1.任取2枚银元分别放在天平的两边.如果天平左右不平衡,则轻的一边就是假银元;如果天平平衡,则进行第二步.2.取下右边的银元,放在一边,然后把剩余的7枚银元依次放在右边进行称量,直到天平不平衡,偏轻的那一枚就是假银元.这种算法最少要称1次,最多要称7次,是不是还有更好的办法,使得称量次数少一些?我们可以采用下面的方法:1.把银元分成3组,每组3枚.2.先将两组分别放在天平的两边.如果天平不平衡,那么假银元就在轻的那一组;如果天平左右平衡,则假银元就在未称的第3组里.3.取出含假银元的那一组,从中任取两枚银元放在天平的两边.如果左右不平衡,则轻的那一边就是假银元;如果天平两边平衡,则未称的那一枚就是假银元.分析理解分析理解•进分析发现,这种算法只需称量两次,这种做法要明显好于前一种做法.当然,这两种方法都具有一般性,同样适用于n枚银元的情形.这是信息论中的一个模型,可以帮助我们找出某些特殊信息.•从以上两个问题中可以看出,同一个问题可能存在着多种算法,其中一些可能要比另一些好.在实际问题和算法理论中,找出好的算法是一项重要的工作.方程近似解算法其算法步骤如下:1.确定有解区间[a,b](f(a)×f(b)0).2.取[a,b]的中点x=(a+b)/2.3.计算函数f(x)在中点处的函数值f((a+b)/2).4.判断函数值f((a+b)/2)是否为0;(1)如果为0,x=(a+b)/2就是方程的解,问题就得到可解决;(2)如果函数值f((a+b)/2)不为0,则分析下列两种情形:①若f(a)×f((a+b)/2)0,则确定新的有解区间为(a,(a+b)/2);xy0baxy=f(x)②若f(a)×f((a+b)/2)0,则确定新的有解区间为((a+b)/2,b).5.判断新的有解区间的长度是否小于精确度:(1)如果新的有解区间长度大于或等于精确度,则在新的有解区间的基础上重复上述步骤;(2)如果新的有解区间长度小于精确度,则取新的有解区间的中点为方程的近似解.例题解析例3求方程f(x)=x3+x2-1=0在[0,1]上的近似解,精确度为0.01.解根据上述分析,可以通过下列步骤求得方程的近似解:1.因为f(0)=-1,f(1)=1,f(0)×f(1)0,则区间[0,1]为有解区间,精确度1-0=10.01;2.取[0,1]的区间中点0.5;3.计算f(0.5)=-0.625;4.由于f(0.5)×f(1)0,可得新的有解区间[0.5,1],精确度1-0.5=0.5〉0.01;5.取[0.5,1]的区间中点0.75;6.计算f(0.75)=-0.01563;7.由于f(0.75)×f(1)0,可得新的有解区间[0.75,1],精确度1-0.75=0.25〉0.01;…当得到新的有解区间[0.75,0.75782]时,由于|0.75782-0.75|=0.0782〈0.01,该区间精确度已满足要求,所以取区间[0.75,0.75782]的中点0.75391,它是方程的一个近似解.课堂练习1.有一把围棋子,5个5个地数,最后余下2个;7个7个地数,最后余下3个;9个9个地数,最后余下4个,请设计一种算法,求出这把围棋子至少有多少个?2.一个大油瓶装8kg油.还有两个空油瓶,一个能装5kg,另一个能装3kg.请设计一种算法,将这8kg油平均分成两份.