本文主要是介绍减治法思想-二分查找图解案例,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
减治法介绍
减治法思想
分治法是将一个大问题划分为若干个子问题,分别求各个子问题,然后把子问题的解进行合并得到原问题的解。
减治法同样是把一个大问题划分为若干个子问题,但是并不是求解所有的子问题,因为原问题的解就在其中一个子问题当中,所以只求解其中一个子问题。
与分治法不同的就在于这个“减”字上,会不断的将原问题的规模减小,直到找到最终的解。
减治法求解过程:
减治法将原问题分解为若干个子问题,并且原问题(规模为n)的解和子问题(规模为 n/2 或 n-1)的解之间存在某种确定的关系:
原问题的解只存在其中一个子问题中。
原问题的解与其中一个子问题的解之间存在某种对应关系。
那么求解过程就可以确定了:
- 将原问题分解为规模较小的子问题
- 找到原问题的解所在的子问题
- 重复第1、2步直到得到子问题的解
- 根据子问题的解,直接得出或计算出原问题的解。
前面蛮力法中使用的案例选择排序其实也是一种减治法,因为其每次都会去除掉一个元素,使问题的规模减一。
减治法案例-二分查找
分析:
二分查找每次计算完数组中间值,与待查找元素对比之后,会使下一次待查找的区间减半,所以二分查找属于比较经典的减治。
过程如下:
- 取有序序列的中间值与待查找值比较。
- 若中间值与待查找值相同则查找成功。
- 若中间值比待查找值小,则在中间记录的左边区间查找。
- 若中间值比待查找值大,则在中间记录的右边区间查找。
- 重复上述1~4过程,直到查找成功或区间无记录。
过程图解:
1、初始化
有序数组:{ -1、0、3、9、11、13、22、27、55、57、60、77 }
待查找元素: 33
初始化:左下标为0(left = 0)、右下标为11(right=11)
2、第一趟比较
计算中间值:
m i d = l e f t + ( r i g h t − l e f t ) / 2 = 0 + ( 11 − 0 ) / 2 = 5 mid = left + (right - left) / 2 = 0 + (11 - 0) / 2 = 5 mid=left+(right−left)/2=0+(11−0)/2=5
减少规模:
n u m s [ 5 ] = 13 < 33 nums[5] = 13 < 33 nums[5]=13<33,带查找值在数组右半区,则 l e f t = m i d + 1 = 5 + 1 = 6 left = mid + 1 = 5 + 1 = 6 left=mid+1=5+1=6。
3、第二躺比较
计算中间值:
m i d = l e f t + ( r i g h t − l e f t ) / 2 = 6 + ( 11 − 6 ) / 2 = 8 mid = left + (right - left) / 2 = 6 + (11 - 6) / 2 = 8 mid=left+(right−left)/2=6+(11−6)/2=8
得到结果:
n u m s [ 8 ] = 33 = 33 nums[8] = 33 = 33 nums[8]=33=33,找到带查找值在数组中的下标8,查找结束。
代码实现:
public static int execute(int[] data, int key) {int leftIndex = 0, rightIndex = data.length - 1;int midIndex = -1;// 当左下标不再小于右下标,说明区间内已经没有元素,即数组内没有待查找元素while (leftIndex < rightIndex) {midIndex = (rightIndex - leftIndex) / 2 + leftIndex;// 找到就直接返回if (data[midIndex] == key) {return midIndex;//中间值小于待查找值,待查找值在数组右半区域} else if (data[midIndex] < key) {leftIndex = midIndex + 1;//中间值大于待查找值,待查找值在数组左半区域} else {rightIndex = midIndex - 1;}}return data[midIndex] == key ? midIndex : -1;}
这篇关于减治法思想-二分查找图解案例的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!