花了半天把二分查找的几种都写了一遍。验证了一下。二分查找的正确编写的关键就是,确保循环的初始、循环不变式能够保证一致。

可以先从循环里面确定循环不变式,然后再推导初始条件,最后根据循环不变式的内容推导出结果。

1. 普通的二分查找

第一版本:

 //first version
int find(int arr[], int n, int target) {
int l = , r = n - ;
while (l <= r) {
int m = l + (r - l) / ;
if (arr[m] == target) return m;
else if (arr[m] < target) {
l = m + ;
} else {
r = m - ;
}
}
return -;
}

循环内部有三次比较,一般来说,相等的操作只需要判断一次,所以最好是放在循环最外面判断。

注意这里因为提到外面判断相等的情况了,那么循环退出条件就可以改为l <r, 在循环中不需要判断l == r的情形。

 //second version
int findLeft(int arr[], int n, int target) {
int l = , r = n - ; // [l, r]
while (l < r) {
int m = l + (r - l) / ; // l < r => m < r
if (arr[m] < target) {
l = m + ; // l <= m => l need to be added by one ensuring l is always increasing; and arr[l] <= target
} else {
r = m; // target <= arr[m], [l, r], m < r => if arr[m] == target, we still decrease r, so what we found is the lower_bound
}
} // [l, r] && r >= l => if (arr[l] == target) => return l;
if (l < n && arr[l] == target) return l;
else return -;
}

2. 查找最左边的值

同上面的version 2.

关键就是,处理相等的情况。由于是要求最左的位置,所以遇到相等时候,仍要把区间往左移。所以这里是用了r=m。(因为m恒小于r)。

3. 查找最右边的值

这里的关键同样是处理相等的情况。要求的是最右的位置,那么遇到相等的时候,还是要把区间右移,所以这里是l=m+1。之所以要加1,就是因为l<=m的,为了确保循环一定能够退出,要加1,才能确保每次循环,区间都在缩小。但是加了1之后,变成了target==arr[m]的时候,l=m+1, 也就是target > arr[l]。所以最终的结果就是l-1。从这里再去推导初始化的时候,l和r的取值。因为区间是[l-1,r),所以l=1,r=n。

 int findRight(int arr[], int n, int target) {
int l = , r = n; // [l - 1, r)
while (l < r) {
int m = l + (r - l) / ;
if (arr[m] > target) { // ensuring target < arr[r] && target > arr[l], [l - 1, r)
r = m; // l < r => m < r, [l - 1, r)
} else {
l = m + ; // l <= m, when target == arr[m], l is still increasing, [l - 1, r), target >= arr[l - 1] = arr[m]
}
} // [l-1, r)
if (l - >= && arr[l - ] == target) return l - ;
else return -;
}

4. 查找倒数第一个比它小的数;

 int findLastSmall(int arr[], int n, int target) {
int l = , r = n - ;
// [l, r]
while (l < r) {
int m = l + (r - l) / ;
if (arr[m] >= target) {
r = m; // target <= arr[r]
} else {
l = m + ; // target > arr[m] => target >= arr[m+1], [l, r]
}
} // target is in [l, r], so the last smaller number is r
if (target > arr[l]) return l;
return l - ;
}

因为循环中确保了target >= arr[l] && target <= arr[r],那么我们要判断target[l]是否等于target,如果等于,返回的是l-1。否则返回的就是l。

5. 查找第一个比它大的数;

int findFirstLarge(int arr[], int n, int target) {
int l = , r = n;
// [l - 1, r)
while (l < r) {
int m = l + (r - l) / ;
if (arr[m] <= target) { // target >= arr[l - 1]
l = m + ; // l is increasing, [l - 1, r)
} else { // target < arr[m]
r = m; // target < arr[r],[l - 1, r)
}
}
// target is in [l - 1, r), so the first larger number is r
return r;
}

这个比较简单,因为循环确定target>=arr[l]&&target < arr[r],那么第一个比target大的数肯定就是arr[r]。

Worst case performance: O(log n)
Best case performance: O(1)
Average case performance: O(log n)
Worst case space complexity: O(1)

今天算是把怎么验证程序的正确性研究了一天了。。。20140923

Algorithm | Binary Search的更多相关文章

  1. [Algorithms] Binary Search Algorithm using TypeScript

    (binary search trees) which form the basis of modern databases and immutable data structures. Binary ...

  2. 【437】Binary search algorithm,二分搜索算法

    Complexity: O(log(n)) Ref: Binary search algorithm or 二分搜索算法 Ref: C 版本 while 循环 C Language scripts b ...

  3. js binary search algorithm

    js binary search algorithm js 二分查找算法 二分查找, 前置条件 存储在数组中 有序排列 理想条件: 数组是递增排列,数组中的元素互不相同; 重排 & 去重 顺序 ...

  4. 2 - Binary Search & LogN Algorithm - Apr 18

    38. Search a 2D Matrix II https://www.lintcode.com/problem/search-a-2d-matrix-ii/description?_from=l ...

  5. 2 - Binary Search & LogN Algorithm

    254. Drop Eggs https://www.lintcode.com/problem/drop-eggs/description?_from=ladder&&fromId=1 ...

  6. [Algorithm] Delete a node from Binary Search Tree

    The solution for the problem can be divided into three cases: case 1: if the delete node is leaf nod ...

  7. [Algorithm] Check if a binary tree is binary search tree or not

    What is Binary Search Tree (BST) A binary tree in which for each node, value of all the nodes in lef ...

  8. [Algorithm] Count occurrences of a number in a sorted array with duplicates using Binary Search

    Let's say we are going to find out number of occurrences of a number in a sorted array using binary ...

  9. leetcode -- Convert Sorted List to Binary Search Tree

    Given a singly linked list where elements are sorted in ascending order, convert it to a height bala ...

随机推荐

  1. 51nod1174(RMQ)

    题目链接:https://www.51nod.com/onlineJudge/questionCode.html#!problemId=1174 题意:中文题诶- 思路:RMQ模板题 关于RMQ: h ...

  2. ubantu16.04+mxnet +opencv+cuda8.0 环境搭建

    ubantu16.04+mxnet +opencv+cuda8.0 环境搭建 建议:环境搭建完成之后,不要更新系统(内核) 转载请注明出处: 微微苏荷 一 我的安装环境 系统:ubuntu16.04 ...

  3. 【Java EE 学习 69 下】【数据采集系统第一天】【实体类分析和Base类书写】

    之前SSH框架已经搭建完毕,现在进行实体类的分析和Base类的书写.Base类是抽象类,专门用于继承. 一.实体类关系分析 既然是数据采集系统,首先调查实体(Survey)是一定要有的,一个调查有多个 ...

  4. TDD学习笔记【五】一隔绝相依性的方式与特性

    前言 在上一篇文章中,提到了如何通过 IoC 的设计,以及 Stub Object 的方式,来独立测试目标对象. 这一篇文章,则要说明有哪些设计对象的方式,可以让测试或需求变更时,更容易转换. 并说明 ...

  5. MySql 获取表的字段名

    mysql安装成功后可以看到已经存在mysql.information_schema和test这个几个数据库,information_schema库中有一个名为COLUMNS的表,这个表中记录了数据库 ...

  6. Beta工作比例(Transcend)

    Beta工作比例 成员 工作 黄志明 10% 洪志兴 10% 李佳恺 17 % 巫振格 17 % 肖承志 10 % 李严 16 % 牛妍辉 stripes 20%

  7. 查看java源码显示source not found

    除了下载的jar包之外,还要下载src源码的zip包. 进项目的Java Build Path里找到你想看源码的jar包 Source Attachment Configuration--Extern ...

  8. java中Map和List初始化的两种方法

    第一种方法(常用方法): //初始化List List<string> list = new ArrayList</string><string>(); list. ...

  9. 转载:安装ie driver和chrome driver

    很多同学在使用webdriver的时候总是忘了安装ie driver和chrome driver, 因此在这里简单介绍一下这2个driver的安装方式. IE driver 在新版本的webdrive ...

  10. 关于android 加载https网页的问题

    我在加载https网页时出现空白, 因此,我就百度一下,可以发现: webView.setWebViewClient(new WebViewClient(){ @Override public voi ...