作者: 佚名 浏览: 日期:2024-06-24
分治,就是把一个复杂的问题分成两个或更多的相同或相似的子问题,再把子问题分成更小的子问题……直到最后子问题可以简单的直接求解,
原问题的解即子问题的解的合并。这个技巧是很多高效算法的基础,如排序算法(快速排序,归并排序),傅立叶变换(快速傅立叶变换)……直接说就是将一个难以直接解决的大问题,分割成一些规模比较小的相同的小问题,以便各个击破,分而治之。
有100枚硬币,其中1枚重量与众不同,是假币,略轻一些。如果用天平秤,请问至少称几次一定能找到这枚假币。
假如我们用传统的逐枚比较法的话,显然至少需要比较50次。
流程如下:
而假如我们采用分治法的话,称量流程如下:
1.将100硬币分成3份,33,33,34。
2.称量1、2份(33,33),若天平平衡,则假币必在另外34枚中。若不平衡,假币在轻的那33枚里。
3.将34枚分为11/11/12枚(或将33枚分成11*3)。
4.称量两组11枚的硬币,若平衡,假币在12枚里(或另外的11枚)若不平衡,
假币在轻的11里。
5.将11(或12)枚分成3/4/4(或4/4/4),称量3/3,方法同上。
6.将剩下的3(或4)分为1/1/1(或1/1),称量1/1,若平衡,则剩下的一枚是假币,若不平衡,轻的是假币。
若还剩4枚,出现1/1平衡,剩下2枚则称量,显然轻的是假币。
而这种方法只要五次就能解决问题。
1.观察可以看到1-2,3-4,5-6步除了硬币的枚数改变了,其他的步骤完全一样。
2.观察发现这是一个子问题的分解过程,100-33-11-3,将一个大问题分解为了容易解决的小问题。
3.可以发现小问题是相互独立的。每一个33枚硬币和其他的并不相互影响。
方便起见,我们用较简单的二分法流程图来具体看一下:
在解决规模为n的问题时,总是先递归地求解a个规模为n/b的子问题,然后在
时间内将子问题的解合并起来,其中a,b,d>0 a,b,d 是一些特定的整数。分治算法的运行时间可以通过公式来计算。
一般的,对于a,b,n>0,且有
成立,
则时间复杂度为:
用比较熟悉的归并排序来分析:
首先有
对于我们熟悉的归并排序符合主定理的第二种情况,有
划分问题:整个问题划分成多个无关联的子问题。
递归求解:递归调用求解各个子问题。
合并问题:合并子问题的解,形成原始问题的解。
用伪代码来具体分析:
其中|P|表示问题P的规模;n0为一阈值,表示当问题P的规模不超过n0时,问题已容易直接解出,不必再继续分解。ADHOC§是该分治法中的基本子算法,用于直接解小规模的问题P。因此,当P的规模不超过n0时直接用算法ADHOC§求解。算法MERGE(y1,y2,…,yk)是该分治法中的合并子算法,用于将P的子问题P1 ,P2 ,…,Pk的相应的解y1,y2,…,yk合并为P的解。
1.该问题的规模缩小到一定的程度就可以容易地解决。(能够使用递归)
2.该问题可以分解为若干个规模较小的相同问题。(可分解)
3.利用该问题分解出的子问题的解可以合并为该问题的解;(可合并)
4.该问题所分解出的各个子问题是相互独立的,即子问题之间不包含公共的子子问题。(非必需)
第1条随着问题规模的减少,问题自然会容易解决。条件2,3是分治的前提。即Divide-and-Conquer的必要条件。
第4条,对于存在公共子问题的问题,使用分治算法会存在重复计算的问题,使用动态规划较为合适。
分治和动态规划有共通也有不同,看如下两个定义。
最优子结构:如果问题的一个最优解中包含了子问题的最优解,则该问题具有最优子机构。
重叠子问题:用来求解原问题的递归算法反复地解同样的子问题,而不是总是在产生新的子问题。对两个子问题来说,如果它们确实是相同的子问题,只是作为不同问题的子问题出现的话,则它们是重叠的。
当子问题相互独立时,能且只能使用分治。存在重叠子问题时,动态规划是更好的算法。
给定n个乱序数字,要求从小到大排序后输出
逐条对应:
1.比较两个数字大小很容易。
2.将n个数字排序和将n/2个数字排序问题相同
3.不存在重叠子问题。因而用分治法是很有效果的。
思路如下:
1.分割:将这个序列分割成一个个已经排好序的子序列(即子序列里只有一个数)
2.比较。
3.归并:将一个个有序的子序列合并成排好序的序列。
用递归的方法来解决。
代码如下
优缺点分析
优点:用分治算法主定理可得时间复杂度为O(nlogn),相同元素的顺序不会颠倒,是稳定排序。
缺点:需要辅助数组,所需空间复杂度为O(n)。
给出一个长度为n的序列A1,A2,A3·····An,求最大连续和,例如,给定(6,-1 , 5, 4,-7), 该序列中的最大和是6 +( - 1)+ 5 + 4 = 14。
基本思路是使用枚举法,三重嵌套循环,时间复杂度是n的三次方。
我们来用分治法解决这个问题
1.划分问题:将序列分成元素个数尽可能相等的两半。
2.递归求解:分别求出位于左半和右半的最佳序列。
3.合并问题:求出起点位于左半,终点位于右半的最大连续和序列,和子问题最优解比较。
如下图所示,从左到右有A、B、C三根柱子,其中A柱子上面有从小叠到大的n个圆盘,现要求将A柱子上的圆盘移到C柱子上去,期间只有一个原则:一次只能移到一个盘子且大盘子不能在小盘子上面,求移动的步骤和移动的次数。
考虑最简单情况n=2,三根柱子分别为A、B、C,此时,在A上有2片金片(假设从小到大依次为1,2)。那么我们的解决步骤可以归结为如下:
1.先把A上1号金片移至B。
2.再把A上2号金片移至C。
3.再把B上1号金片移至C。
考虑n增大的情况:
1.先把A上从1到n-1号金片移至B。
2.再把A上n号金片移至C。
3.再把B上从1到n-1号金片移至C。
可以看到递归的影子,要解决n个,先解决n-1个,可以分治为n=2的最简单情况并加以解决。