[转]编程珠玑第五章二分搜索(折半查找)之java实现
http://blog.csdn.net/hwe_xc/article/details/51813080
二分搜索又称为折半查找,用来高效快速的解决如下问题:
我们需要确定排序后的数组x[0..n-1] 中是否包含目标元素t。 说的准确一点,就是已知
n>=0且x[0]<=x[1]<=x[2]…<=x[n-1],当n=0时数组为空,t与x中元素的数据类型相同。答案存在一个整数中,为-1则不在x数组中,否则返回具体位置0~n-1。
此问题看起来是个很小的问题,其实也不容易。据说第一篇二分搜索论文是1946年发表,但是对于完全没有错误的二分搜索程序却是在1962年才出现,用了16年的时间。可想而知,要想写出一个基本没有错误的程序不是那么简单的事儿。
下面列出此程序需要注意的地方
注意点
1. 一定要对数组进行排序
2. 如果被搜索的项不在数组中,需要返回 -1
3. 如果被搜索项多次出现,是否有特殊需求,比如需要返回第一次出现的位置
代码实现
/**
* 在排序好的数组中通过二分搜索t的位置(对于t重复出现不要求找出特定位置)
*
* @param a
* 排序好的数组
* @param t
* 被查找项
* @return
*/
public int binarySearch(int a[], int t) {
int i = 0, j = a.length - 1;
int result = -1;
while (i <= j) {
// 取得中间值
int m = (i + j) / 2;
// 被查找项大于中间值
if (t > a[m]) {
// 查找 m + 1 ~ j 区间
// 注意:一定是 m + 1开始,如果是m,有可能会陷入死循环之中
i = m + 1;
// 被查找项小于中间值
} else if (t < a[m]) {
// 查找 i ~ m - 1 区间
// 注意:一定是 m - 1结束,如果是m,有可能会陷入死循环之中
j = m - 1;
} else {
// 被查找项等于中间值, 直接返回
result = m;
break;
}
}
return result;
}
上面的代码是二分查找的一般形式,对于在t重复出现情况下没要求的话,完全够用了。如果需要返回第一次t出现的位置,我们又该如何写,请看下面代码:
public static int binarySearchFirstIndex(int a[], int t) {
int i = 0, j = a.length - 1;
int result = -1;
while (i <= j) {
// 取得中间值
int m = (i + j) / 2;
// 被查找项大于中间值
if (t > a[m]) {
// 查找 m + 1 ~ j 区间
// 注意:一定是 m + 1开始,如果是m,有可能会陷入死循环之中
i = m + 1;
// 被查找项小于中间值
} else if (t < a[m]) {
// 查找 i ~ m - 1 区间
// 注意:一定是 m - 1结束,如果是m,有可能会陷入死循环之中
j = m - 1;
} else {
// 被查找项等于中间值, 结果等于中间值
result = m;
// 附加判断中间值的左边值是否等于t
// 注意m > 0,如果不判断可能引起数组下表越界
if (m > 0 && t == a[m - 1]) {
// 继续查找
j = m - 1;
} else {
break;
}
}
}
return result;
}
可以看到,上面代码有一点小小的改进,会去判断中间值是否等于其左边的值,如果等于继续查找。对于查找最后一个位置,原理类似。
[转]编程珠玑第五章二分搜索(折半查找)之java实现的更多相关文章
- [书籍翻译] 《JavaScript并发编程》第五章 使用Web Workers
本文是我翻译<JavaScript Concurrency>书籍的第五章 使用Web Workers,该书主要以Promises.Generator.Web workers等技术来讲解Ja ...
- 安卓权威编程指南 - 第五章学习笔记(两个Activity)
学习安卓编程权威指南第五章的时候自己写了个简单的Demo来加深理解两个Activity互相传递数据的问题,然后将自己的学习笔记贴上来,如有错误还请指正. IntentActivityDemo学习笔记 ...
- 折半查找(java)(边学习边更新)
---恢复内容开始--- class ArrayTest3 { public static void main(String[] args) { //int [] arr=new int[]{54,4 ...
- 算法笔记_026:折半查找(Java)
目录 1 问题描述 2 解决方案 2.1 递归法 2.2 迭代法 1 问题描述 首先,了解一下何为折半查找?此处,借用<算法设计与分析基础>第三版上一段文字介绍: 2 解决方案 2.1 递 ...
- 【读书笔记】《编程珠玑》第一章之位向量&位图
此书的叙述模式是借由一个具体问题来引出的一系列算法,数据结构等等方面的技巧性策略.共分三篇,基础,性能,应用.每篇涵盖数章,章内案例都非常切实棘手,解说也生动有趣. 自个呢也是头一次接触编程技巧类的书 ...
- 《编程珠玑》第二章 aha算法
A题:给定一个最多包含40亿个随机排列的32位整数的顺序文件,找出一个不在文件中的32位整数. 1.在文件中至少存在这样一个数? 2.如果有足够的内存,如何处理? 3.如果内存不足,仅可以用文件来进行 ...
- 《Python核心编程》第五章:数字
PS:[笔记+代码+图片]在GitHub上持续更新,欢迎star:https://github.com/gdouchufu/Core-Python-Programming 本章大纲 介绍Python支 ...
- 《UNIX网络编程》 -- 第五章
str_cli 和 str_echo 函数 需要先弄清楚 3.9 readn.writen 和 readline 函数 str_cli void str_cli(FILE *fp, int sockf ...
- Windows核心编程 第十五章 在应用程序中使用虚拟内存
第1 5章 在应用程序中使用虚拟内存 Wi n d o w s提供了3种进行内存管理的方法,它们是: • 虚拟内存,最适合用来管理大型对象或结构数组. • 内存映射文件,最适合用来管理大型数据流(通常 ...
随机推荐
- vc++基础班[27]---实现一个简单的任务管理器
因为任务管理器中涉及到进程的枚举操作,所以把两节课的知识点合并到一起来讲! ①.设计界面.以及列表控件变量的绑定: ②.列表控件样式的指定: m_TaskList.SetExtendedSty ...
- 通过URL传递中文参数的乱码处理
环境:web.xml中配置了 <filter> <filter-name>encodingFilter</filter-name> <filter-class ...
- android studio定时器
1.超时 CountDownTimer第一个参数超时时间,第二个参数多久执行一次onTick(), 到达设定的超时时间执行onFinsh(),cancel取消超时计数,start重新开始(从零开始). ...
- $Django redis内存数据库 (知识回顾cmd切换目录)
知识小回顾 #切换盘 C:\Users\WangDong>f: F:\> #切换文件 F:\>cd redis F:\redis> #返回上一级 F:\DJ\dj8>cd ...
- Laravel 5.2控制器--RESTful 资源控制器
laravel的资源控制器 一.创建资源控制器 在命令行上,创建一个新的资源控制器 php artisan make:controller TestController --resource 然后在你 ...
- 解决FTPClient上传文件为空,显示0字节
JAVA使用FTPClient上传文件时总是为空,而使用FileZilla客户端时却不会. 后来查了下资料,FTP服务器有被动模式和主动模式.(具体查另外资料) 在JAVA中将FTPClient设置为 ...
- Caffeine缓存
在本文中,我们来看看 Caffeine — 一个高性能的 Java 缓存库. 缓存和 Map 之间的一个根本区别在于缓存可以回收存储的 item. 回收策略为在指定时间删除哪些对象.此策略直接影响缓存 ...
- 将CSDN内容移过来
将CSND主要博客内容移过来 1.Linux rpm.yum.ssh.apache.Samba等讲解 https://blog.csdn.net/weixin_38834998/article/det ...
- 1283: 骨牌铺方格(zzuli)
Problem Description 在2×n的一个长方形方格中,用一个1× 2的骨牌铺满方格,输入n ,输出铺放方案的总数.例如n=3时,为2× 3方格,骨牌的铺放方案有三种,如下图: Input ...
- C#概念总结(五)
1.C#特性 (Attribute) 特性适用于在运行时传递程序中的各中元素(比如 类.方法.结构.枚举.组件)的行为信息额声明性标签,可以通过使用特性向程序添加声明性信息,一个声明标签是通过放置在他 ...