二分法
有序数组中找一个值
我们直接来看题目
要求在一个有序的数组中找一个数的下标 要你在这个数组中找到该值 并且返回它的下标 如果找不到就返回-1
如果说我们使用遍历的方式遍历一整个数组的话那么此时我们的时间复杂度就是O(N)了
但是这样子就浪费了有序这一个条件 这个时候就可以考虑使用我们的二分法
我们假设该有序数组如下 我们要找的元素是4
我们找到该数组最左边的下标和最右边的下标 将它们相加之后除以2 在该题目中我们找到的下标就是4
之后我们比较五下标对应的元素和我们要找的值之间对应的关系 之后二分法的精髓就来了
因为该数组为有序数组所以说:
如果中间下标的值大于我们要找的值 是不是从中间位置开始整个数组的左半边我们都可以舍弃了
如果中间下标的值小于我们要找的值 是不是从中间位置开始整个数组的右半边我们都可以舍弃了
也就是说我们通过一次操作就排除了一半的选项 这就是二分法的精髓 也是二分法时间复杂度O(logN)的原因
我们下面可以看看代码如何表示
#include <vector> #include <iostream> using namespace std; int binsearch(vector<int> v , int num) { int left = 0; int right = v.size() - 1; while(left <= right) { int mid = (left + right) / 2 ; if (v[mid] == num) { return mid; } else if (v[mid] > num) { right = mid - 1; } else { left = mid + 1; } } return -1; } int main() { vector<int> v1 = {0 , 1 , 4, 6, 7, 8, 9, 10 , 11 ,12}; int ret = binsearch(v1 , 4); cout << ret << endl; return 0; }
代码很简单 但是这里也有一点小细节需要注意
我们的 left + right
在数组很大的情况下有可能会产生溢出的情况
所以说我们可以采用这种写法
// mid = (left + right) /2 防止溢出 mid = left + (right - left) / 2;
该代码运行结果如下
下标确实是2 符合我们的预期
寻找有序数组中大于等于某个数的位置
给你一个有序数组 要求你找到从某个值开始 后面的所有值都大于等于我们的key值
该题目也是一个典型的二分法的应用
我们假设有序数组如下 要求我们从某个下标开始所有值都大于等于三
还是和上面的思路一样 我们找出该数组的左右下标之后找出中间位置
如果该下标对应的值大于等于我们key值 那么我们就记录下该下标 之后排除掉右边所有的值
如果该下标对应的值小于我们的key值 那么我们就排除掉左边所有的值
和第一个题目不同的是 我们在做该题时一次二分并不能找到我们的答案 必须要二分到最后一个值才能确定我们的答案是多少 (因为可能会有重复的值)
这也是我们记录下标的意义 每次找到符合要求的下标我们就更新它
代码如下
#include <iostream> using namespace std; #include <vector> void binsearch(vector<int> v , int num) { int left = 0; int right = v.size() -1 ; int index = 0; while(left <= right) { int mid = left + ((right - left) / 2); if (v[mid] >= num) { index = mid; right = mid -1; } else { left = mid + 1; } } cout << "find the index is:" << index << endl; } int main() { vector<int> v1 = {1 , 2 , 3, 3, 3, 4, 5, 5, 6, 7, 8, 9, 9, 9}; binsearch(v1 , 3); return 0; }
运行结果如下
我们发现确实符合我们的预期
总结:
从这个题目中我们应该能够加深对于二分法的理解 它的特点在于能够一次性排除一半的值
通过这个特点我们能够很快速的在有序数组中找到某一个特定的位置
无序数组中寻找一个局部最小值
给你一个无序数组 (数组元素大于3)该数组中每个元素都不相等 要求你找到一个局部最小值
我们这里首先给出一个局部最小值的概念
如果这个值比它前一个位置和后一个位置的值都小 那么我们就认为这个值是一个局部最小值
如果是起始位置和终止位置我们就只需要比较它和后一个(前一个)位置就可以
明白局部最小值的概念之后我们肯定会先考虑两个特殊位置 最前面和最后面的位置
如果说这其中有一个位置是局部最小值我们直接返回就好了
如果说它们都不是一个局部最小值 那么该数组的大小排序应该是一个这样子的状态
中间是一个什么状态我们不知道 但是根据起始和终止位置我们能够推断的是 中间肯定会存在一个局部最小值
因为只有存在一个局部最小值该曲线到最后才会是一个上升的状态 否则就会是一直下降了
那么我们就可以通过二分法来找到这其中一个局部最小值
我们通过二分法找到中间位置之后该位置和相邻两个位置的关系只可能是这三种
如果是第一种情况 那么我们就可以直接返回该下标的位置 因为这就是一个局部最小值
如果是情况二 我们可以画出这样子的图来理解
通过这张图我们就可以推断出右边肯定会存在一个局部最小值 可以继续二分
同理如果是情况三
我们可以推断出左边肯定会存在一个局部最小值 可以继续二分
#include <iostream> using namespace std; #include <vector> void binsearch(vector<int>& v1) { int left = 0; int right = v1.size() -1; // left if (v1[1] > v1[0]) { cout << "find the min num:" << v1[left] << "index:" << left << endl; return; } // right if (v1[right -1 ] > v1[right]) { cout << "find the min num:" << v1[right] << "index:" << right << endl; return; } while(left <= right) { int mid = (left + right) / 2; if (v1[mid] < v1[mid - 1]) { if (v1[mid] < v1[mid + 1]) { cout << "find the min num:" << v1[mid] << " index: " << mid << endl; return; } else { left = mid + 1; } } else { right = mid - 1; } } } int main() { vector<int> v1 = {3 , 1, 5, 6, 7, 5, 7, 8, 1, 9}; binsearch(v1); return 0; }
运行结果如下
我们可以发现1 确实是一个局部最小值
同样的 从这个题目中 我们可以更加深入的理解二分法
不一定有序才可以二分 二分的精髓在于想办法一次性排除掉一半的数据