地球人都知道“二分查找”,方法也非常简单,但是你能不能在10分钟内写出一个没有bug的程序呢?

知易行难,自己动手写一下试一试吧。

public class BinarySearch {
    public static int search(int [] A,int target,int a, int b)
    {
        int  middle = (a+b)/2;
        if(a>b)
            return -1;
        else if (A[middle]==target)
            return middle;
        else if (A[middle]< target)
            return search(A,target,middle+1,b);
        else
            return search(A,target,a,middle-1);
    }
    public static void main(String[] args) {
        // TODO Auto-generated method stub
        int A [] = {2,4,6,8,10};
        System.out.println( BinarySearch.search(A,1 , 0, A.length-1)  );
        System.out.println( BinarySearch.search(A,2 , 0, A.length-1)  );
        System.out.println( BinarySearch.search(A,3 , 0, A.length-1)  );
        System.out.println( BinarySearch.search(A,4 , 0, A.length-1)  );
        System.out.println( BinarySearch.search(A,5 , 0, A.length-1)  );
        System.out.println( BinarySearch.search(A,6 , 0, A.length-1)  );
        System.out.println( BinarySearch.search(A,7 , 0, A.length-1)  );
        System.out.println( BinarySearch.search(A,8 , 0, A.length-1)  );
        System.out.println( BinarySearch.search(A,9 , 0, A.length-1)  );
        System.out.println( BinarySearch.search(A,10 , 0, A.length-1) );
        System.out.println( BinarySearch.search(A,11 , 0, A.length-1) );
    }
}

上面的代码看似是没有任何问题了,但是呢,其实还是有一个很微妙的bug,这个bug发生在:

int  middle = (a+b)/2;

这一行。想一下,如果a+b超出了Int型的最大值了呢???所以呢,修正这个bug应该这样:
public class BinarySearch {
    public static int search(int [] A,int target,int a, int b)
    {
        int  middle = a+(b-a)/2;
        if(a>b)
            return -1;
        else if (A[middle]==target)
            return middle;
        else if (A[middle]< target)
            return search(A,target,middle+1,b);
        else
            return search(A,target,a,middle-1);
    }
    public static void main(String[] args) {
        // TODO Auto-generated method stub
        int A [] = {2,4,6,8,10};
        System.out.println( BinarySearch.search(A,1 , 0, A.length-1)  );
        System.out.println( BinarySearch.search(A,2 , 0, A.length-1)  );
        System.out.println( BinarySearch.search(A,3 , 0, A.length-1)  );
        System.out.println( BinarySearch.search(A,4 , 0, A.length-1)  );
        System.out.println( BinarySearch.search(A,5 , 0, A.length-1)  );
        System.out.println( BinarySearch.search(A,6 , 0, A.length-1)  );
        System.out.println( BinarySearch.search(A,7 , 0, A.length-1)  );
        System.out.println( BinarySearch.search(A,8 , 0, A.length-1)  );
        System.out.println( BinarySearch.search(A,9 , 0, A.length-1)  );
        System.out.println( BinarySearch.search(A,10 , 0, A.length-1) );
        System.out.println( BinarySearch.search(A,11 , 0, A.length-1) );
    }
}

查找一个元素是否存在,自然分为“存在”和“不存在”两种情况。在思考二分查找的时候,先思考“存在”的情况,设计算法,然后再去考虑“不存在”的情况。

如果同时思考“存在”和“不存在”两种情况,那么很容易被绕晕。

《算法导论》习题2.3-5 二分搜索 Binary Search的更多相关文章

  1. 【LeetCode-面试算法经典-Java实现】【096-Unique Binary Search Trees(唯一二叉搜索树)】

    [096-Unique Binary Search Trees(唯一二叉搜索树)] [LeetCode-面试算法经典-Java实现][全部题目文件夹索引] 原题 Given n, how many s ...

  2. 数据结构与算法--二分搜索(binary search)

    前言 之前面试准备秋招,重新翻起了<编程之美>.在第三章节看到了一道关于二分搜索的讨论,觉得有许多细节是自己之前也没怎么特别注意地方,比如二分搜索的初始条件,转化.终止条件之类的. 问题 ...

  3. LeetCode算法题-Lowest Common Ancestor of a Binary Search Tree

    这是悦乐书的第197次更新,第203篇原创 01 看题和准备 今天介绍的是LeetCode算法题中Easy级别的第59题(顺位题号是235).给定二叉搜索树(BST),找到BST中两个给定节点的最低共 ...

  4. 二分搜索 - Binary Search

    二分搜索是一种在有序数组中寻找目标值的经典方法,也就是说使用前提是『有序数组』.非常简单的题中『有序』特征非常明显,但更多时候可能需要我们自己去构造『有序数组』.下面我们从最基本的二分搜索开始逐步深入 ...

  5. 算法学习记录-查找——折半查找(Binary Search)

    以前有个游戏,一方写一个数字,另一方猜这个数字.比如0-100内一个数字,看谁猜中用的次数少. 这个里面用折半思想猜会大大减少次数. 步骤:(加入数字为9) 1.因为数字的范围是0-100,所以第一次 ...

  6. 《算法导论》习题2.3-6 改进的InsertSort

    InsertSort中有关键的一步是把当前元素A[i]插入到已经排好序的A[1,i-1]的合适的位置上,在原始的InsertSort算法中, 采用的是从后往前一步一步查找的方法,习题2.3-6要求利用 ...

  7. [置顶] 《算法导论》习题解答搬运&&学习笔记 索引目录

    开始学习<算法导论>了,虽然是本大部头,可能很难一下子看完,我还是会慢慢地研究的. 课后的习题和思考有些是很有挑战性的题目,我等蒻菜很难独立解决. 然后发现了Google上有挺全的algo ...

  8. (搬运)《算法导论》习题解答 Chapter 22.1-1(入度和出度)

    (搬运)<算法导论>习题解答 Chapter 22.1-1(入度和出度) 思路:遍历邻接列表即可; 伪代码: for u 属于 Vertex for v属于 Adj[u] outdegre ...

  9. 算法导论课后习题解答 第一部分 练习1.1-1->1.1-5

    很高兴能和大家一起共同学习算法导论这本书.笔者将在业余时间把算法导论后面的题解以博文的形式展现出来希望能得到大家的支持谢谢.如果有可能我会做一些教学视频免费的供大家观看. 练习题选自算法导论中文第三版 ...

随机推荐

  1. ReactiveCocoa & FRP & MVVM

    Functional Reactive Programming(以下简称FRP)是一种响应变化的编程范式.先来看一小段代码 a = 2 b = 2 c = a + b // c is 4 b = 3 ...

  2. eclipse背景颜色修改插件color theme

    对于长时间编码的程序猿来说,这个插件可以说是福音了. 安装方式有两种: 1.可以直接在eclipse的eclipse marketplace 那里搜索安装 2.可以在Help->Install ...

  3. 视频录制SurfaceView

    package com.bw.videorecorder; import java.io.File;import java.io.IOException; import android.media.M ...

  4. Python科学计算学习一 NumPy 快速处理数据

    1 创建数组 (1) array(boject, dtype=None, copy=True, order=None, subok=False, ndmin=0) a = array([1, 2, 3 ...

  5. 转:jmeter实践

    本文主要介绍性能测试中的常用工具jmeter的使用方式,以方便开发人员在自测过程中就能自己动手对系统进行自动压测和模拟用户操作访问请求.最后还用linux下的压测工具ab做了简单对比. 1.      ...

  6. JavaBean,List,Map转成json格式

    public class User { private String username; private String password; public String getUsername() { ...

  7. startActivityForResult相关的

    在Fragment里调用startActivityForResult,界面回来后,会先走其宿主Activity的onActivityResult方法,再走Fragment的. 在Fragment里面嵌 ...

  8. Oracle 使用sql创建表空间及用户

    create tablespace OrcalDBNamedb datafile 'C:\OracleDBDirc\OrcalDBNamedb.dbf' size 300m; 创建用户create u ...

  9. stock 仓位

    别胆大求多,不轻信盲从.抓住几只自己长期关注并看好的股票.将这几只股票选为自选股,而其他股,不管是机构推荐还是股评荐股,都要谨慎,不轻易听从. 巧用“三三制”,根据趋势控制仓位.当不知道是在涨还是在跌 ...

  10. [转]使用openssl库实现RSA、AES数据加密

    openssl是可以很方便加密解密的库,可以使用它来对需要在网络中传输的数据加密.可以使用非对称加密:公钥加密,私钥解密.openssl提供了对RSA的支持,但RSA存在计算效率低的问题,所以一般的做 ...