ImageVerifierCode 换一换
格式:DOCX , 页数:13 ,大小:1.30MB ,
资源ID:6276283      下载积分:3 金币
快捷下载
登录下载
邮箱/手机:
温馨提示:
快捷下载时,用户名和密码都是您填写的邮箱或者手机号,方便查询和重复下载(系统自动生成)。 如填写123,账号就是123,密码也是123。
特别说明:
请自助下载,系统不会自动发送文件的哦; 如果您已付费,想二次下载,请登录后访问:我的下载记录
支付方式: 支付宝    微信支付   
验证码:   换一换

加入VIP,免费下载
 

温馨提示:由于个人手机设置不同,如果发现不能下载,请复制以下地址【https://www.bdocx.com/down/6276283.html】到电脑端继续下载(重复下载不扣费)。

已注册用户请登录:
账号:
密码:
验证码:   换一换
  忘记密码?
三方登录: 微信登录   QQ登录  

下载须知

1: 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。
2: 试题试卷类文档,如果标题没有明确说明有答案则都视为没有答案,请知晓。
3: 文件的所有权益归上传用户所有。
4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
5. 本站仅提供交流平台,并不能对任何下载内容负责。
6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

版权提示 | 免责声明

本文(分治算法讲解.docx)为本站会员(b****5)主动上传,冰豆网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。 若此文所含内容侵犯了您的版权或隐私,请立即通知冰豆网(发送邮件至service@bdocx.com或直接QQ联系客服),我们立即给予删除!

分治算法讲解.docx

1、分治算法讲解分治算法一:基本概念(分而治之)分治就是把一个复杂的问题分成两个或更多的相同或相似的子问题,再把子问题分成更小的子问题直到最后子问题可以简单的直接求解,原问题的解即子问题的解的合并。比如:二分查找,归并排序,快速排序,树的遍历等等任何一个可以用计算机求解的问题所需的计算时间都与其规模有关。问题的规模越小,越容易直接求解,解题所需的计算时间也越少。例如,对于n个元素的排序问题,当n=1时,不需任何计算。n=2时,只要作一次比较即可排好序。n=3时只要作3次比较即可,。而当n较大时,问题就不那么容易处理了。要想直接解决一个规模较大的问题,有时是相当困难的。二:基本思想 分治设计思想:将

2、一个大的问题,分解成一个个小的,相同类型的问题,然后逐个击破各个小问题,最后将小问题逐步合并,得到最终的解。(可能会用到递归,大问题里包含小问题,找到规律然后解决) 分治基本策略:对于一个规模为n的问题,若该问题可以容易地解决(比如说规模n较小)则直接解决,否则将其分解为k个规模较小的子问题,这些子问题互相独立且与原问题形式相同,递归地解这些子问题,然后将各子问题的解合并得到原问题的解。 三:分治使用情况 1) 该问题的规模缩小到一定的程度就可以容易地解决2) 该问题可以分解为相同类型的小问题(前提)3) 利用该问题分解出的子问题的解可以合并为该问题的解;(关键) 4) 该问题所分解出的各个子

3、问题是相互独立的,即子问题之间不包含公共的子子问题。 第三条特征是关键,能否利用分治法完全取决于问题是否具有第三条特征,如果具备了第一条和第二条特征,而不具备第三条特征,则可以考虑用贪心法或动态规划法。 四:基本步骤 (1)划分:把问题的实例划分成子问题 (2)求解:若是子问题比较简单,就直接解决,否则递归求解子问题 (3)合并:合并子问题的解得到原问题的解课前引导:一:分段函数 例如:高中数学中的分段函数也是类似分治思想的体现,如看图求y关于x的表达式。一个分段函数,反映的是x与y的关系,简单来说,就是在R的范围内将y的表达式表示出来,那么这时候利用分治的思想,将R区间划分为小区间,然后分别

4、求出各个小区间的表达式,最后合并起来,完成y关于x的表达式的求解二:大整数乘法123 345 678 * 3 = 370 037 034在这里我们可以这样写:123 * 3 = 369345 * 3 = 1035678 * 3=2034组合在一起是369 1035 2034。对比发现,当使用千进制的时候结果变成了370037034首先他满足:第一条件:分解到一定小规模的时候可以解决第二条件:每个小规模都具有最佳子结构(在变量范围内,可以用来表示)第三条件:每个小问题可以通过合并在一起形成大问题的解第四条件:每个小问题相互独立专题一:分治算法之二分查找思考题:找假币: 有一堆个数为32的硬币,和

5、一个天平,已知其中有一个假币,且假币比真硬币轻,找出这个假币1.普通方法:两两比较,轻的那个是假币,最多比较16次2.二分法:将硬币分为两份,假币在轻的那份中,然后继续分,直到找出假币,最多用5次 哪种方法好?课题:二分查找(折半查找)知识目标:理解二分查找算法的概念以及执行过程。 重点:掌握二分查找算法的常规写法以及递归写法。1.边界错误造成的问题 2.死循环3.溢出I. 算法介绍: 二分查找又称折半查找,优点是比较次数少,查找速度快,平均性能好;其缺点是要求待查表为有序表,且插入删除困难。因此,折半查找方法适用于不经常变动而查找频繁的有序列表。II. 思路分析:二分查找的基本思想是:(1)

6、先确定一组顺序排列的数据存储到数组中,输入要查找的数据(2)将数组元素的中值与查找的数据相比较,如果两者相等,则子函数返回相应的结果后终止;(3)否则利用中间位置缩小数据查找的范围。如果中间位置的数组元素大于查找数值,则进一步查找中值之前的数组元素,否则进一步查找中值之后的数组元素。(4)重复上述过程,直到在数组中找到相同的数字。(5)若在数组中找不到这个数据,则显示查找不成功III. 算法框架: 按照分治算法三步骤,将二分算法作如下介绍:(1)二分算法代码设计模式: /arr表示要进行二分查找的顺序排列对象数组,low表示数组下标的最小值,high表示数组下表的最大值,key表示要查找的元素

7、 int erfen(int arr,int low,int high,int key) 如果数组下标的最小值大于最大值 则返回结果为-1至主函数;/表明在数组中不存在要查找的元素 确定数组的中间位置mid 若查找的元素等于数组的中间元素,则进行相应的步骤 若查找的元素大于数组(指定范围内)的中间元素 则将查找范围缩小至数组(指定范围内)中间元素右边; 若查找的元素小于数组指定范围内的中间元素 则将查找范围缩小至数组(指定范围内)中间元素左边; 例题1:输入一个整数n,然后按升序输入n个整数,将它们存入数组a中,再输入一个数x,然后在数组中查找x,如果找到,输出相应的最小下标,否则,输出“No

8、t Found”. 普通写法:#includeusing namespace std;int main() int i,s100,n; cinn; for(i=0;isi; int x; cinx; for(i=0;in;i+) if(si=x) coutiendl; break; if(i=n) coutNot foundendl; return 0;二分写法:#includeusing namespace std;void erfen(int a,int n,int key) int low=0,high=n-1,mid; while(low=high) mid=(low+high)/2;

9、 if(amid=key) cout这个数的下标是:midkey) high=mid-1; else low=mid+1; if(lowhigh) coutNot Found!n; for(i=0;iai; cinkey; erfen(a,n,key); return 0;二分递归:#include using namespace std;int search(int a,int left,int right,int key) if(leftright) coutNot found!endl; exit(0); else int middle=(left+right)/2; if (amidd

10、le=key) return middle; else if(keyn; for(i=0;iai; left=0; right=n-1; cinx; cout这个数的下标是:search(a,left,right,x)endl;专题二:分治算法之归并排序归并排序是分治算法的一个非常典型的应用。归并排序原理:归并排序具体工作原理如下(假设序列共有n个元素):将序列每相邻两个数字进行归并操作(merge),形成floor(n/2)个序列,排序后每个序列包含两个元素将上述序列再次归并,形成floor(n/4)个序列,每个序列包含四个元素重复步骤2,直到所有元素排序完毕归并操作:归并操作(merge)

11、,也叫归并算法,指的是将两个顺序序列合并成一个顺序序列的方法。如设有数列6,202,100,301,38,8,1初始状态:6,202,100,301,38,8,1第一次归并后:6,202,100,301,8,38,1,比较次数:3;第二次归并后:6,100,202,301,1,8,38,比较次数:4;第三次归并后:1,6,8,38,100,202,301,比较次数:4;总的比较次数为:3+4+4=11;归并基本算法:输入两个整数,作为两个数组的长度,输入两个按升序排列好的数组,将两个已排序的数组合并后存放在另一个数组中,且合并后的数组也是有序排列(要求不能合并后再排序),再输出合并后的数组。【

12、样例输入】4 51 2 3 45 6 7 8 9【样例输出】1 2 3 4 5 6 7 8 9代码:#includeusing namespace std;int main() int m,n,i,j,k=0,a100,b100,c200; cinmn; for(i=0;iai; for(j=0;jbj; i=0;j=0; while(im&jn)/当数组a和数组b都没有完全赋值到数组c中时 if(aibj)/如果a数组里的元素比b数组的小 ck=ai;/就把数组a的元素赋给数组c i+;k+;/且将数组a和数组c的下标都往后移一位 else/要是数组b的元素比数组a的元素大时 ck=bj;/

13、就把数组b的元素赋值到数组c中 j+;k+;/且将数组b和数组c的下标往后移一位 if(i=m)/当数组a已经被完全赋值到数组c中 while(jn)/当数组b还没有完全赋值 ck=bj;/此时,只需要把b数组中剩余的数全部赋值到数组c接下去的位置上 j+;k+; else/当数组b已经被完全赋值到数组c中 while(im)/当数组a还没有完全赋值 ck=ai;/此时,只需要把a数组中剩余的数全部赋值到数组c接下去的位置上 i+;k+; for(i=0;im+n;i+) coutci ; 归并函数:所涉及知识过多,目前只需要了解思想【例题三】设有n=2k个运动员要进行网球循环赛。现要设计一个

14、满足以下要求的比赛日程表: (1)每个选手必须与其他n-1个选手各赛一次; (2)每个选手一天只能参赛一次; (3)循环赛在n-1天内结束请按此要求将比赛日程表设计成有n行和n-1列的一个表。在表中的第i行,第j列处填入第i个选手在第j天所遇到的选手。其中1in,1jn-1。假设有8位参赛选手,8个选手的比赛日程表如下图:【思路】按分治的实现过程,可以先找到上面所示日程表的规律,即对角线相等,那么所要完成的操作就是对角线填充。实现过程:(1)用一个for循环输出日程表的第一行 for(int i=1;i=N;i+) a1i = i (2)然后定义一个m值,m初始化为1,m用来控制每一次填充表格

15、时i(i表示行)和j(j表示列)的起始填充位置。 (3)用一个for循环将问题分成几部分,对于k=3,n=8,将问题分成3大部分,第一部分为,根据已经填充的第一行,填写第二行,第二部分为,根据已经填充好的第一部分,填写第三四行,第三部分为,根据已经填充好的前四行,填写最后四行。for (ints=1;s=k;s+) N/=2; (4)用一个for循环对中提到的每一部分进行划分for(int t=1;t=N;t+)对于第一部分,将其划分为四个小的单元,即对第二行进行如下划分 同理,对第二部分(即三四行),划分为两部分,第三部分同理。 (5)最后,根据以上for循环对整体的划分和分治法的思想,进行每一个单元格的填充。填充原则是:对角线填充 for(int i=m+1;i=2*m;i+) /i控制行 for(int j=m+1;j=2*m;j+) /j控制列 aij+(t-1)*m*2= ai-mj+(t-1)*m*2-m;/*右下角的值等于左上角的值 */ aij+(t-1)*m*2-m =ai-mj+(t-1)*m*2;/*左下角的值等于右上角的值 */ 实例过程: (1)由初始化的第一行填充第二行 (2)由s控制的第一部分填完。然后是s+,进行第二部分的填充 (3)最后是第三部分的填充

copyright@ 2008-2022 冰豆网网站版权所有

经营许可证编号:鄂ICP备2022015515号-1