NOIP基础算法枚举、递推和递归ppt课件.ppt
《NOIP基础算法枚举、递推和递归ppt课件.ppt》由会员分享,可在线阅读,更多相关《NOIP基础算法枚举、递推和递归ppt课件.ppt(115页珍藏版)》请在三一办公上搜索。
1、NOIP基础算法综合,第一部分,枚举策略,一、枚举法的基本思想,枚举法的基本思想是根据提出的问题枚举所有可能状态,并用问题给定的条件检验哪些是需要的,哪些是不需要的。能使命题成立,即为其解。枚举结构:循环+判断语句。,二、枚举法的条件:,虽然枚举法本质上属于搜索策略,但是它与后面讲的回溯法有所不同。因为适用枚举法求解的问题必须满足两个条件:可预先确定每个状态的元素个数n;状态元素a1,a2,an的可能值为一个连续的值域。,三、枚举法的框架结构,设ai1状态元素ai的最小值;aik状态元素ai的最大值(1in),即a11a1a1k,a21a2a2k, ai1aiaik,an1anankfor a
2、1a11 to a1k do for a2a21 to a2k do for aiai1 to aik do for anan1 to ank do if 状态(a1,ai,an)满足检验条件 then 输出问题的解;,枚举法的优点由于枚举算法一般是现实生活中问题的“直译”,因此比较直观,易于理解;由于枚举算法建立在考察大量状态、甚至是穷举所有状态的基础上,所以算法的正确性比较容易证明。,枚举法的缺点枚举算法的效率取决于枚举状态的数量以及单个状态枚举的代价,因此效率比较低。,四、枚举法的优缺点,“直译”枚举:直接根据题意设定枚举对象、范围和约束条件。 注意认真审题,不要疏漏任何条件,例题1:砝
3、码称重(noip1996),【问题描述】设有1g、2g、3g、5g、10g、20g的砝码各若干枚(其总重=1000),求用这些砝码能称出不同的重量个数。【文件输入】输入1g、2g、3g、5g、10g、20g的砝码个数。【文件输出】输出能称出不同重量的个数。【样例输入】1 1 0 0 0 0【样例输出】3,【分析】根据输入的砝码信息,每种砝码可用的最大个数是确定的,而且每种砝码的个数是连续的,能取0到最大个数,所以符合枚举法的两个条件,可以使用枚举法。枚举时,重量可以由1g,2g,20g砝码中的任何一个或者多个构成,枚举对象可以确定为6种重量的砝码,范围为每种砝码的个数。判定时,只需判断这次得到
4、的重量是新得到的,还是前一次已经得到的,即判重。由于重量=1000g,所以,可以开一个a1001的数组来判重。,核心参考代码:,readln(a,b,c,d,e,f)for c1:=0 to a do /1g砝码的个数 for c2:=0 to b do /2g砝码的个数 for c3:=0 to c do /3g砝码的个数 for c4:=0 to d do /5g砝码的个数 for c5:=0 to e do /10g砝码的个数 for c6:=0 to f do /20g砝码的个数 begin sum:=0; for i:=1 to 6 do sum:=sum+ci*wi; asum:=
5、1; /标记 end;for i:=1 to 1000 do if ai=1 then inc(num); /统计不同重量的个数Writeln(num);,【问题描述】给你n根火柴棍,你可以拼出多少个形如“A+B=C”的等式?等式中的A、B、C是用火柴棍拼出的整数(若该数非零,则最高位不能是0)。用火柴棍拼数字0-9的拼法如图所示:注意: 1. 加号与等号各自需要两根火柴棍 2. 如果AB,则A+B=C与B+A=C视为不同的等式(A、B、C0) 3. n根火柴棍必须全部用上【输入】输入文件matches.in共一行,又一个整数n(n24)。【输出】输出文件matches.out共一行,表示能拼
6、成的不同等式的数目。,例题2:火柴棒等式(NOIP2008),例题2:火柴棒等式(NOIP2008),【问题简述】给你n(n=24)根火柴棒,叫你拼出 “A + B = C”这样的等式,求方案数。,【思路点拨】本题主要考查对枚举法的掌握,可以枚举A和B的取值,考查等式是否刚好用了24根火柴棒。1S的时限对枚举的范围有所要求,必须要仔细分析A和B的取值。,例题2:火柴棒等式(NOIP2008),本题最多24根火柴,等号和加号共用4根火柴,所以A,B,C这3个数字需用20根火柴。我们考查A和B的最大的取值可能:09这10个数字所用的火柴数为6,2,5,5,4,5,6,3,7,6,很明显数字1用的火
7、柴棒最少只要2根,不妨让B为1,那么A和C最多可以使用18根火柴,而C=A,满足条件的A的最大取值为1111。所以枚举A和B的范围是从01111。,为了加快速度,可以将0到2222的所有整数需要的火柴棒数目提前算好保存在数组中。,五、枚举算法的优化,枚举算法的时间复杂度:状态总数*单个状态的耗时。,提取有效信息;减少重复计算;将原问题化为更小的问题;根据问题的性质进行截枝;引进其他算法,【例题3】给你n(n=105)个整数,然后要有m (m=105)个询问。每个询问两个整数i和j,问第i个数字到第j个数字所有数字之和。,【朴素算法】 readln(n); for i:=1 to n do re
8、ad(ai); for i:=1 to m do begin read(x,y); sum:=0; for j:=x to y do sum:=sum+aj; writeln(sum); end;时间复杂度为:O(nm),【优化算法】先递推计算出si=si-1+ai,再回答询问情况; readln(n); for i:=1 to n do统计求和 begin read(ai); si:=si-1+ai; end; for i:=1 to m do begin read(x,y); writeln(sy-sx-1); end;,【例题4】对于给定的N*M的矩形,在其中找一个R*C的权值最大的区域
9、, 1=N,M=1,000。,【算法一】:以每一个格子为左上角枚举R*C的区域,并求出它的权值之和。最后取其中的最大值。此算法包含4重循环。 时间复杂度:O(N4),【算法二】:我们设Si,j表示以(1,1)为左上角,(i,j)为右下角区域的权值之和,那么我们以(i,j)为右下角的R*C区域权值之和的计算公式为: Areai,j=Si,j+Si-R,j-C-Si-R,j-Si,j-C其中Si,j的计算公式为: Si,j=valuei,j+Si-1,j+Si,j-1-Si-1,j-1 你可以随手画图出来,很容易即可证明上面两个式子。最后取Area中的最大值即可。 时间复杂度:O(N2),【例题5
10、】最大连续子序列的和,【问题描述】给你一个有n(n=100000)个整数的序列,要求你求出其中最大连续子序列的和。,【算法1】枚举起始位置,结束位置,计算中间的和。时间复杂度O(n3)。 maxx:=a1; for i:=1 to n do for j:=i to n do begin sum:=0; for k:=i to j do sum:=sum+ak; if summaxx then maxx:=sum; end;,【算法2:优化的枚举O(n2) s0:=0; for i:=1 to n do si:=si-1+ai;/初始化求和 maxx:=a1; for i:=1 to n do
11、for j:=i to n do if sj-si-1maxx then maxx:=sj-si-1;时间复杂度:预处理+主程序=O(n+n2)=O(n2),n=5000,【算法3】分治O(nlogn)、 最大连续子序列的位置有三种可能: 完全处于序列的左半; 完全处于序列的右半; 跨越序列中间;,【算法4】DPO(n) 1、阶段和状态: 以第i个数结尾的最大连续子序列的和,只存在两种选择: 情形1:只包含ai 情形2:包含ai和以ai-1结尾的最大连续和序列 故设fi表示以ai结尾的最大连续子序列的和 2、状态转移方程: 转移方程:fi=maxai,fi-1+ai(2=i=n) 初始化:f1
12、=a1 Answer=maxfi|1=i=n,【例题6】最大子矩阵问题,【问题描述】给定一个二维的数组(含正数或负数),请从中找出和最大的子矩阵。例如:,1、“直译”枚举过程For x1:=1 to n do/枚举矩形左上角(x1,y1) for y1:=1 to n do for x2:=1 to n do/枚举矩形右下角(x2,y2) for y2:=1 to n do /考察状态左上角为(x1,y1)右下角为(x2,y2)内矩形的元素之和; begin sum:=0; for x:=x1 to x2 do/计算当前矩形内元素的和 for y:=y1 to y2 do sum:=sum+a
13、x,y; if sumbest then best:=sum;/调整最优解 end;这个算法相当粗糙,枚举状态的费用为O(n6),2、从减少重复计算入手 有刚才一维情况可以推广到二维,在统计左上角为(x1,y1)右下角为(x2,y2)内矩形的元素之和时,我们同样可以先初始化,计算出左上角为(1,1),右下角为(x,y)内矩形的元素之和sxy。 for i:=1 to n do /枚举矩形右下角,求和 for j:=1 to n do begin read(ai,j); si,j:=si-1,j+si,j-1-si-1,j-1+ai,j; end;对于状态左上角为(x1,y1),右下角为(x2,
14、y2)内矩形的元素之和,可以改为: sum:=sx2,y2-sx1-1,y2-sx2,y1-1+sx1-1,y1-1; if sumbest then best:=sum; /调整最优解由于先进行了预处理,整个算法的时间复杂度降为O(n4),3、提取恰当的信息容易观察到,最大子矩阵问题是最大连续子序列和问题的提升,即将一条线换成一个面,将一维问题提升到二维问题。所以我们计算最大子矩阵的方法就是将一行行的数进行累加以求得最大值。但是还有一个问题,那就是应该如何高效地存储矩阵?我们可以想到:在一个一维的数列中,设数组bi表示从第1个元素到第i个元素的和,则如果想要求第i个元素到第j个元素的和,只需
15、要计算bj-bi-1的值就行了。由此推广到二维矩阵,设bi,j表示矩阵第j列前i个元素的和,ai,j表示元素数据,则压缩存储: for i:=1 to n do for j:=1 to n do begin read(ai,j);bi,j:=bi-1,j+ai,j;因此,我们可以使用三重循环求出所有的矩形值,即枚举起始行i和终止行j,压缩子矩形成为一行,变成一维求最大子段和问题。 即tk=max(tk-1,0)+bj,k-bi-1,k;时间复杂度为O(n3),0 -2 -7 0 2 -6 2-4 1 -4 1-1 8 0 -2,0 -2 -7 0 0 -13 25 1 -17 34 9 -17
16、 1,核心代码 sum:=-99999999; /置初值 for i:=1 to n do /阶段:起始行 begin for j:=i to n do /状态:结束行 begin t1:=bj,1-bi-1,1; /初始化第1列的值 for k:=2 to n do /决策:第几列 begin if tk-10 then tk:=tk-1+bj,k-bi-1,k else tk:=bj,k-bi-1,k; if tksum then sum:=tk; end; end; end; writeln(sum);,六、局部枚举,例题7:求第一、第二、第三最短路问题,例题8:新年好,重庆城里有n个车
17、站,m条双向公路连接其中的某些车站。每两个车站最多用一条公路直接相连,从任何一个车站出发都可以经过一条或多条公路到达其他车站,但不同的路径需要花费的时间可能不同。在一条路上花费的时间等于路径上所有公路需要的时间之和。佳佳的家在车站1,他有五个亲戚,分别住在车站a,b,c,d,e。过年了,他需要从自己的家出发,拜访每个亲戚(顺序任意),给他们送去节日的祝福。怎样走,才需要最少的时间?数据范围:n(n=50,000),m(m=100,000),算法框架为:,(1)用6次计算单源最短路算法,即枚举这6个点,每个点都做为源点求单源最短路。(2)枚举(a,b,c,d,e)这5个结点的全排列,找到一个使得
18、总路程长度最短的方案。 这一题中的边数远小于n2,所以复杂度也只有nlogn+m。,第二部分,递推策略,一、引例:Fibonacci数列,Fibonacci数列的代表问题是由意大利著名数学家Fibonacci于1202年提出的“兔子繁殖问题”(又称“Fibonacci问题”)。问题: 一个数列的第0项为0,第1项为1,以后每一项都是前两项的和,这个数列就是著名的裴波那契数列,求裴波那契数列的第N项。,解答,由问题,可写出递推方程,算法: f0:=1; f1:=2; for i:=2 to n do fi:=fi1+fi2;,总结,从这个问题可以看出,在计算裴波那契数列的每一项目时,都可以由前两
19、项推出。这样,相邻两项之间的变化有一定的规律性,我们可以将这种规律归纳成如下简捷的递推关系式:Fn=g(Fn-1),这就在数的序列中,建立起后项和前项之间的关系。然后从初始条件(或是最终结果)入手,按递推关系式递推,直至求出最终结果(或初始值)。很多问题就是这样逐步求解的。对一个试题,我们要是能找到后一项与前一项的关系并清楚其起始条件(或最终结果),问题就可以递推了,接下来便是让计算机一步步了。让高速的计算机从事这种重复运算,真正起到“物尽其用”的效果。,二、递推概念,给定一个数的序列H0,H1,Hn,若存在整数n0,使当nn0时,可以用等号(或大于号、小于号)将Hn与其前面的某些项Hn(0i
20、n)联系起来,这样的式子就叫做递推关系。如何建立递推关系递推关系有何性质如何求解递推关系,三、解决递推问题的一般步骤,建立递推关系式 确定边界条件 递推求解,四、递推的两种形式,顺推法和倒推法,五、递推的应用分类,一般递推问题 组合计数类问题 一类博弈问题的求解 动态规划问题的递推关系,例题1:faibonacci数列,【问题描述】已知faibonacci数列的前几个数分别为0,1,1,2,3,5,编程求出此数列的第n项。(n=60),(一)递推的应用(一般递推问题),思考:当n=109时,如何求解?,(一)递推的应用(一般递推问题),例题2:输出杨辉三角的前N行 【问题描述】输出杨辉三角的前
21、N行(N10)。【文件输入】输入只有一行,包括1个整数N(N10)。【文件输出】输出只有N行。【样例输入】3 【样例输出】 1 1 1 1 2 1,递推方程:fi,j=fi-1j-1+fi-1,j边界条件:fi,1=1;fi,i=1,(一)递推的应用(一般递推问题),例题3 : Hanoi塔问题 . Hanoi塔由n个大小不同的圆盘和三根木柱a,b,c组成。开始时,这n个圆盘由大到小依次套在a柱上,如图1所示。要求把a柱上n个圆盘按下述规则移到c柱上: (1)一次只能移一个圆盘; (2)圆盘只能在三个柱上存放; (3)在移动过程中,不允许大盘压小盘。 问将这n个盘子从a柱移动到c柱上,总计需要
22、移动多少个盘次?,分析,2,1,3,当n=1时:AC当n=2时:AB,AC,BC当n=3时:,分析,设f(n)为n 个盘子从1柱移到3柱所需移动的最少盘次。当n=1时,f(1)=1。当n=2时,f(2)=3。以此类推,当1柱上有n(n2)个盘子时,我们可以利用下列步骤:第一步:先借助3柱把1柱上面的n-1个盘子移动到2柱上,所需的 移动次数为f(n-1)。第二步:然后再把1柱最下面的一个盘子移动到3柱上,只需要1 次盘子。第三步:再借助1柱把2柱上的n-1个盘子移动到3上,所需的移动 次数为f(n-1)。由以上3步得出总共移动盘子的次数为:f(n-1)+1+ f(n-1)。所以:f(n)=2
23、f(n-1)+1 hn=2hn-1+1 =2n-1 边界条件:h1=1,【扩展1】: Hanoi双塔问题,【扩展2】:四塔问题,【问题分析】令fi表示四个柱子时,把i个盘子从原柱移动到目标柱所需的最少移动次数。,j,第一步:先把1柱上的前j个盘子移动到另外其中一个非目标柱(2或3柱均可,假设移到2柱)上,此时3和4柱可以作为中间柱。移动次数为:fj。,第二步:再把原1柱上剩下的i-j个盘子在3根柱子(1、3、4)之间移动,最后移动到目标柱4上,因为此时2柱不能作为中间柱子使用,根据三柱问题可知,移动次数为:2(i-j)-1。,第三步:最后把非目标柱2柱上的j个盘子移动到目标柱上,次数为:fj。
24、,i,通过以上步骤我们可以初步得出: fi = 2*fj+2(i-j)-1,j可取的范围是1=ji,所以对于不同的j,得到的fi可能是不同的,本题要求最少的移动次数。,fi=min2*fj+2(i-j)-1,其中1=ji,【扩展3】:m塔问题,【问题分析】 设F(m,n)为m根柱子,n个盘子时移动的最少次数: 1、先把1柱上的前j个盘子移动到另外其中一个除m柱以外的非目标柱上,移动次数为:fm, j; 2、再把原1柱上剩下的n-j个盘子在m-1根柱子之间移动,最后移动到目标柱m上,移动次数为:fm-1, n-j; 3、最后把非目标柱上的j个盘子移动到目标柱没柱上,移动次数为:fm, j。,F(
25、m,n)=min2*F(m,j)+F(m-1,n-j) (1=jn),j,(一)递推的应用(一般递推问题),例题4:数的计数 【问题描述】我们要求找出具有下列性质数的个数(包含输入的自然数n),先输入一个自然数n(n1000),然后对此自然数按照如下方法进行处理: (1)、不作任何处理; (2)、在它的左边加上一个自然数,但该自然数不能超过原数的一半; (3)、加上数后,继续按此规则进行处理,直到不能再加自然数为止;,方法1:用递推用hn表示自然数n所能扩展的数据个数,则: h1=1,h2=2,h3=2,h4=4,h5=4, h6=6,h7=6,h8=10,h9=10。分析上数据,可得递推公式
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- NOIP基础算法 枚举、递推和递归ppt课件 NOIP 基础 算法 枚举 递归 ppt 课件
链接地址:https://www.31ppt.com/p-1377264.html