跳转到内容

User:深鸣/二分查找算法

本页使用了标题或全文手工转换
维基百科,自由的百科全书
深鸣/二分查找算法
二分查找过程示意,目标值为7
概况
類別搜索算法
資料結構数组
复杂度
平均時間複雜度
最坏时间复杂度
最优时间复杂度
空間複雜度
最佳解
相关变量的定义

二分查找[1](英語:binary search,台湾作二分搜尋[2],大陆又作二分搜索[3][a]是用于查找有序数组英语Sorted array中目标值位置的搜索算法[9][10]

In computer science, binary search, also known as half-interval search, logarithmic search, or binary chop, is a search algorithm that finds the position of a target value within a sorted array. Binary search compares the target value to the middle element of the array. If they are not equal, the half in which the target cannot lie is eliminated and the search continues on the remaining half, again taking the middle element to compare to the target value, and repeating this until the target value is found. If the search ends with the remaining half being empty, the target is not in the array.

Binary search runs in logarithmic time in the worst case, making comparisons, where is the number of elements in the array.[b][11] Binary search is faster than linear search except for small arrays. However, the array must be sorted first to be able to apply binary search. There are specialized data structures designed for fast searching, such as hash tables, that can be searched more efficiently than binary search. However, binary search can be used to solve a wider range of problems, such as finding the next-smallest or next-largest element in the array relative to the target even if it is absent from the array.

There are numerous variations of binary search. In particular, fractional cascading speeds up binary searches for the same value in multiple arrays. Fractional cascading efficiently solves a number of search problems in computational geometry and in numerous other fields. Exponential search extends binary search to unbounded lists. The binary search tree and B-tree data structures are based on binary search.

算法

[编辑]

二分查找适用于有序数组。二分查找首先比较数组中间的元素与目标值。如果目标值与该元素匹配,则返回该元素在数组中的位置;如果目标值小于该元素,则在数组较小的那一半中继续查找;如果目标值大于该元素,则在数组较大的那一半中继续查找。通过这种方法,每次迭代都能将搜索范围缩小一半。[12]

过程

[编辑]

给定一个包含个元素的数组,其中的值或记录分别为,且满足。假设目标值为。下面的子程序使用二分查找来寻找在数组中的索引。[12]

  1. 如果,则搜索失败并终止。
  2. (中间元素的位置)为向下取整值,即不大于的最大整数。
  3. 如果,则取,并返回步骤2。
  4. 如果,则取,并返回步骤2。
  5. 如果,则搜索完成,返回

这个迭代过程使用两个变量来跟踪搜索边界。该过程可以用伪代码表示如下,其中变量名和类型与上文相同,floor下取整函数unsuccessful表示搜索失败时的特定返回值:[12]

二分查找过程
function binary_search(A, n, T) is
    L := 0
    R := n − 1
    while L ≤ R do
        m := floor((L + R) / 2)
        if A[m] < T then
            L := m + 1
        else if A[m] > T then
            R := m − 1
        else:
            return m
    return unsuccessful

也可取向上取整值。如此所做,若目标值在数组中出现多次,结果可能会有所不同。

替代过程

[编辑]

上述过程中,每次迭代都会检查中间元素()是否等于目标值()。而在其他一些实现中,仅剩下一个元素(即)时,才会执行这项检查,这样每次迭代时就无需检查。这种方式的比较循环更快,因为每次迭代少了一次比较,但平均只需要多一次迭代。[13]

赫尔曼·博滕布鲁赫英语Hermann Bottenbruch于1962年首次发表了省略此检查的实现。[13][14]

  1. 时,
    1. (中间元素的位置)为向上取整值,即不小于的最小整数。
    2. 如果,取
    3. 否则说明,取
  2. 现在,搜索完成。如果,返回。否则,搜索失败并终止。

该版本的伪代码如下,其中ceil是上取整函数:

function binary_search_alternative(A, n, T) is
    L := 0
    R := n − 1
    while L != R do
        m := ceil((L + R) / 2)
        if A[m] > T then
            R := m − 1
        else:
            L := m
    if A[L] = T then
        return L
    return unsuccessful

重复元素

[编辑]

即使数组中存在重复元素,算法可能返回任意一个与目标值相等的索引。例如,如果要搜索的数组为,且目标值为,那么算法返回第4个(索引为 3)或第5个(索引为4)元素都是正确的。常规过程通常返回第4个元素(索引为3),但并不总是返回第一个重复项(考虑数组,这时依然返回第4个元素)。然而,有时需要找到目标值在数组中重复出现的最左侧或最右侧的元素。在上述例子中,第4个元素是值为4的最左侧元素,而第5个元素是值为4的最右侧元素。上述的替代过程总是会返回最右侧元素的索引(如果该元素存在的话)。[14]

查找最左侧元素的过程

[编辑]

要查找最左边的元素,可以使用以下过程:[15]

  1. 时,
    1. (中间元素的位置)为的向下取整值,即不大于的最大整数。
    2. 如果,取
    3. 否则说明,取
  2. 返回

如果,那么是等于的最左侧元素。即使不在数组中,也是在数组中的排序位置,即数组中小于的元素数量。

该版本的伪代码如下,其中floor是下取整函数:

function binary_search_leftmost(A, n, T):
    L := 0
    R := n
    while L < R:
        m := floor((L + R) / 2)
        if A[m] < T:
            L := m + 1
        else:
            R := m
    return L

查找最右侧元素的过程

[编辑]

要查找最右边的元素,可以使用以下过程:[15]

  1. 时,
    1. (中间元素的位置)为的向下取整值,即不大于的最大整数。
    2. 如果,取
    3. 否则说明,取
  2. 返回

如果,那么是等于的最右侧元素。即使不在数组中,也是数组中大于的元素数量。

该版本的伪代码如下,其中floor是下取整函数:

function binary_search_rightmost(A, n, T):
    L := 0
    R := n
    while L < R:
        m := floor((L + R) / 2)
        if A[m] > T:
            R := m
        else:
            L := m + 1
    return R - 1

近似匹配

[编辑]
二分查找可计算近似匹配。上图中,目标值本身不在数组中,但通过近似匹配,能计算出其排序位置、前驱、后继、最近邻

上述过程仅用于精确匹配,即找到目标值的位置。然而,由于二分查找在有序数组上进行,所以很容易扩展它以执行近似匹配。例如,二分查找可以用来计算给定值的排序位置(即比它小的元素的数量)、前驱(前一个较小的元素)、后继(下一个较大的元素)、最近邻范围查询英语Range query (computer science)(查找两个值之间的元素数量)可以通过查询两次排序位置来完成。[16]

  • 查询排序位置可以使用查找最左侧元素的过程来完成。程序的返回值即为小于目标值的元素数量。[16]
  • 查询前驱可以通过查询排序位置来执行。如果目标值的排序位置为,那么其前驱的位置为[17]
  • 对于后继查询,可以查找最右侧元素。如果得到的结果为,那么目标值的后继位置就是[17]
  • 目标值的最近邻是其前驱或后继之一,取决于哪个位置更接近。
  • 区间查询也很简单。[17]一旦知道了两个值的位置,区间内大于等于第一个值且小于第二个值的元素数量就是两个位置之差。考虑到是否需要将区间的端点包含在内,以及数组中是否包含与端点匹配的元素,其结果可能会增加或减少1。[18]

性能

[编辑]
表示二分查找的。这里要查找的数组是,目标值是
最坏情况下,搜索到达树的最深层;而最好情况是目标值正好为中间元素

就比较次数而言,可以将二分查找的过程放在二叉树上执行,以分析其性能。树的根节点是数组的中间元素。左半部分的中间元素是根节点的左子节点,右半部分的中间元素是根节点的右子节点,其余部分以类似方式构建。搜索过程从根节点开始,根据目标值是小于还是大于当前节点的值来选择遍历左子树还是右子树。[11][19]

最坏情况下,二分查找需要比较次。此时搜索会达到树的最深层,并且对于任何二分查找,树的层数总为。若目标元素不在数组中,可能会发生最坏情况:若可以表示为2的某次幂减1,那么查找过程总会遍历到最深层,一定会发生最坏情况;否则,搜索过程可能会在倒数第二层中止,此时比较了次,比最坏情况少一次。[20]

平均情况下,当目标元素在数组中时,二分查找的比较次数是(假设每个元素被搜索的概率相等),近似于;若目标元素不在数组中,二分查找的比较次数平均为(假设范围内及范围外的元素被搜索的概率相等)。[19]

最好情况下,即目标值正好是数组的中间元素,二分查找在一次比较后就能返回其位置。[21]

In terms of iterations, no search algorithm that works only by comparing elements can exhibit better average and worst-case performance than binary search. The comparison tree representing binary search has the fewest levels possible as every level above the lowest level of the tree is filled completely.[c] Otherwise, the search algorithm can eliminate few elements in an iteration, increasing the number of iterations required in the average and worst case. This is the case for other search algorithms based on comparisons, as while they may work faster on some target values, the average performance over all elements is worse than binary search. By dividing the array in half, binary search ensures that the size of both subarrays are as similar as possible.[19]

空间复杂度

[编辑]

Binary search requires three pointers to elements, which may be array indices or pointers to memory locations, regardless of the size of the array. Therefore, the space complexity of binary search is in the word RAM model of computation.

平均情况的推导

[编辑]

The average number of iterations performed by binary search depends on the probability of each element being searched. The average case is different for successful searches and unsuccessful searches. It will be assumed that each element is equally likely to be searched for successful searches. For unsuccessful searches, it will be assumed that the intervals between and outside elements are equally likely to be searched. The average case for successful searches is the number of iterations required to search every element exactly once, divided by , the number of elements. The average case for unsuccessful searches is the number of iterations required to search an element within every interval exactly once, divided by the intervals.[19]

成功的搜索

[编辑]

In the binary tree representation, a successful search can be represented by a path from the root to the target node, called an internal path. The length of a path is the number of edges (connections between nodes) that the path passes through. The number of iterations performed by a search, given that the corresponding path has length , is counting the initial iteration. The internal path length is the sum of the lengths of all unique internal paths. Since there is only one path from the root to any single node, each internal path represents a search for a specific element. If there are elements, which is a positive integer, and the internal path length is , then the average number of iterations for a successful search , with the one iteration added to count the initial iteration.[19]

Since binary search is the optimal algorithm for searching with comparisons, this problem is reduced to calculating the minimum internal path length of all binary trees with nodes, which is equal to:[22]

For example, in a 7-element array, the root requires one iteration, the two elements below the root require two iterations, and the four elements below require three iterations. In this case, the internal path length is:[22]

The average number of iterations would be based on the equation for the average case. The sum for can be simplified to:[19]

Substituting the equation for into the equation for :[19]

For integer , this is equivalent to the equation for the average case on a successful search specified above.

失败的搜索

[编辑]

Unsuccessful searches can be represented by augmenting the tree with external nodes, which forms an extended binary tree. If an internal node, or a node present in the tree, has fewer than two child nodes, then additional child nodes, called external nodes, are added so that each internal node has two children. By doing so, an unsuccessful search can be represented as a path to an external node, whose parent is the single element that remains during the last iteration. An external path is a path from the root to an external node. The external path length is the sum of the lengths of all unique external paths. If there are elements, which is a positive integer, and the external path length is , then the average number of iterations for an unsuccessful search , with the one iteration added to count the initial iteration. The external path length is divided by instead of because there are external paths, representing the intervals between and outside the elements of the array.[19]

This problem can similarly be reduced to determining the minimum external path length of all binary trees with nodes. For all binary trees, the external path length is equal to the internal path length plus .[22] Substituting the equation for :[19]

Substituting the equation for into the equation for , the average case for unsuccessful searches can be determined:[19]

另一过程的性能

[编辑]

Each iteration of the binary search procedure defined above makes one or two comparisons, checking if the middle element is equal to the target in each iteration. Assuming that each element is equally likely to be searched, each iteration makes 1.5 comparisons on average. A variation of the algorithm checks whether the middle element is equal to the target at the end of the search. On average, this eliminates half a comparison from each iteration. This slightly cuts the time taken per iteration on most computers. However, it guarantees that the search takes the maximum number of iterations, on average adding one iteration to the search. Because the comparison loop is performed only times in the worst case, the slight increase in efficiency per iteration does not compensate for the extra iteration for all but very large .[d][23][24]

运行时间和缓存使用

[编辑]

In analyzing the performance of binary search, another consideration is the time required to compare two elements. For integers and strings, the time required increases linearly as the encoding length (usually the number of bits) of the elements increase. For example, comparing a pair of 64-bit unsigned integers would require comparing up to double the bits as comparing a pair of 32-bit unsigned integers. The worst case is achieved when the integers are equal. This can be significant when the encoding lengths of the elements are large, such as with large integer types or long strings, which makes comparing elements expensive. Furthermore, comparing floating-point values (the most common digital representation of real numbers) is often more expensive than comparing integers or short strings.[25]

On most computer architectures, the processor has a hardware cache separate from RAM. Since they are located within the processor itself, caches are much faster to access but usually store much less data than RAM. Therefore, most processors store memory locations that have been accessed recently, along with memory locations close to it. For example, when an array element is accessed, the element itself may be stored along with the elements that are stored close to it in RAM, making it faster to sequentially access array elements that are close in index to each other (locality of reference). On a sorted array, binary search can jump to distant memory locations if the array is large, unlike algorithms (such as linear search and linear probing in hash tables) which access elements in sequence. This adds slightly to the running time of binary search for large arrays on most systems.[25]

二分搜索与其他方案

[编辑]

Sorted arrays with binary search are a very inefficient solution when insertion and deletion operations are interleaved with retrieval, taking time for each such operation. In addition, sorted arrays can complicate memory use especially when elements are often inserted into the array.[26] There are other data structures that support much more efficient insertion and deletion. Binary search can be used to perform exact matching and set membership (determining whether a target value is in a collection of values). There are data structures that support faster exact matching and set membership. However, unlike many other searching schemes, binary search can be used for efficient approximate matching, usually performing such matches in time regardless of the type or structure of the values themselves.[27] In addition, there are some operations, like finding the smallest and largest element, that can be performed efficiently on a sorted array.[16]

线性搜索

[编辑]

Linear search is a simple search algorithm that checks every record until it finds the target value. Linear search can be done on a linked list, which allows for faster insertion and deletion than an array. Binary search is faster than linear search for sorted arrays except if the array is short, although the array needs to be sorted beforehand.[e][29] All sorting algorithms based on comparing elements, such as quicksort and merge sort, require at least comparisons in the worst case.[30] Unlike linear search, binary search can be used for efficient approximate matching. There are operations such as finding the smallest and largest element that can be done efficiently on a sorted array but not on an unsorted array.[31]

二叉树

[编辑]
Binary search trees are searched using an algorithm similar to binary search.

A binary search tree is a binary tree data structure that works based on the principle of binary search. The records of the tree are arranged in sorted order, and each record in the tree can be searched using an algorithm similar to binary search, taking on average logarithmic time. Insertion and deletion also require on average logarithmic time in binary search trees. This can be faster than the linear time insertion and deletion of sorted arrays, and binary trees retain the ability to perform all the operations possible on a sorted array, including range and approximate queries.[27][32]

However, binary search is usually more efficient for searching as binary search trees will most likely be imperfectly balanced, resulting in slightly worse performance than binary search. This even applies to balanced binary search trees, binary search trees that balance their own nodes, because they rarely produce the tree with the fewest possible levels. Except for balanced binary search trees, the tree may be severely imbalanced with few internal nodes with two children, resulting in the average and worst-case search time approaching comparisons.[f] Binary search trees take more space than sorted arrays.[34]

Binary search trees lend themselves to fast searching in external memory stored in hard disks, as binary search trees can be efficiently structured in filesystems. The B-tree generalizes this method of tree organization. B-trees are frequently used to organize long-term storage such as databases and filesystems.[35][36]

散列表

[编辑]

For implementing associative arrays, hash tables, a data structure that maps keys to records using a hash function, are generally faster than binary search on a sorted array of records.[37] Most hash table implementations require only amortized constant time on average.[g][39] However, hashing is not useful for approximate matches, such as computing the next-smallest, next-largest, and nearest key, as the only information given on a failed search is that the target is not present in any record.[40] Binary search is ideal for such matches, performing them in logarithmic time. Binary search also supports approximate matches. Some operations, like finding the smallest and largest element, can be done efficiently on sorted arrays but not on hash tables.[27]

集合

[编辑]

A related problem to search is set membership. Any algorithm that does lookup, like binary search, can also be used for set membership. There are other algorithms that are more specifically suited for set membership. A bit array is the simplest, useful when the range of keys is limited. It compactly stores a collection of bits, with each bit representing a single key within the range of keys. Bit arrays are very fast, requiring only time.[41] The Judy1 type of Judy array handles 64-bit keys efficiently.[42]

For approximate results, Bloom filters, another probabilistic data structure based on hashing, store a set of keys by encoding the keys using a bit array and multiple hash functions. Bloom filters are much more space-efficient than bit arrays in most cases and not much slower: with hash functions, membership queries require only time. However, Bloom filters suffer from false positives.[h][i][44]

其他数据结构

[编辑]

There exist data structures that may improve on binary search in some cases for both searching and other operations available for sorted arrays. For example, searches, approximate matches, and the operations available to sorted arrays can be performed more efficiently than binary search on specialized data structures such as van Emde Boas trees, fusion trees, tries, and bit arrays. These specialized data structures are usually only faster because they take advantage of the properties of keys with a certain attribute (usually keys that are small integers), and thus will be time or space consuming for keys that lack that attribute.[27] As long as the keys can be ordered, these operations can always be done at least efficiently on a sorted array regardless of the keys. Some structures, such as Judy arrays, use a combination of approaches to mitigate this while retaining efficiency and the ability to perform approximate matching.[42]

其他形式

[编辑]

统一二分查找

[编辑]
Uniform binary search stores the difference between the current and the two next possible middle elements instead of specific bounds.

Uniform binary search stores, instead of the lower and upper bounds, the difference in the index of the middle element from the current iteration to the next iteration. A lookup table containing the differences is computed beforehand. For example, if the array to be searched is [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11], the middle element () would be 6. In this case, the middle element of the left subarray ([1, 2, 3, 4, 5]) is 3 and the middle element of the right subarray ([7, 8, 9, 10, 11]) is 9. Uniform binary search would store the value of 3 as both indices differ from 6 by this same amount.[45] To reduce the search space, the algorithm either adds or subtracts this change from the index of the middle element. Uniform binary search may be faster on systems where it is inefficient to calculate the midpoint, such as on decimal computers.[46]

指数搜索

[编辑]
Visualization of exponential searching finding the upper bound for the subsequent binary search

Exponential search extends binary search to unbounded lists. It starts by finding the first element with an index that is both a power of two and greater than the target value. Afterwards, it sets that index as the upper bound, and switches to binary search. A search takes iterations before binary search is started and at most iterations of the binary search, where is the position of the target value. Exponential search works on bounded lists, but becomes an improvement over binary search only if the target value lies near the beginning of the array.[47]

插值搜索

[编辑]
Visualization of interpolation search using linear interpolation. In this case, no searching is needed because the estimate of the target's location within the array is correct. Other implementations may specify another function for estimating the target's location.

Instead of calculating the midpoint, interpolation search estimates the position of the target value, taking into account the lowest and highest elements in the array as well as length of the array. It works on the basis that the midpoint is not the best guess in many cases. For example, if the target value is close to the highest element in the array, it is likely to be located near the end of the array.[48]

A common interpolation function is linear interpolation. If is the array, are the lower and upper bounds respectively, and is the target, then the target is estimated to be about of the way between and . When linear interpolation is used, and the distribution of the array elements is uniform or near uniform, interpolation search makes comparisons.[48][49][50]

In practice, interpolation search is slower than binary search for small arrays, as interpolation search requires extra computation. Its time complexity grows more slowly than binary search, but this only compensates for the extra computation for large arrays.[48]

分数级联

[编辑]
In fractional cascading, each array has pointers to every second element of another array, so only one binary search has to be performed to search all the arrays.

Fractional cascading is a technique that speeds up binary searches for the same element in multiple sorted arrays. Searching each array separately requires time, where is the number of arrays. Fractional cascading reduces this to by storing specific information in each array about each element and its position in the other arrays.[51][52]

Fractional cascading was originally developed to efficiently solve various computational geometry problems. Fractional cascading has been applied elsewhere, such as in data mining and Internet Protocol routing.[51]

推广到图表

[编辑]

Binary search has been generalized to work on certain types of graphs, where the target value is stored in a vertex instead of an array element. Binary search trees are one such generalization—when a vertex (node) in the tree is queried, the algorithm either learns that the vertex is the target, or otherwise which subtree the target would be located in. However, this can be further generalized as follows: given an undirected, positively weighted graph and a target vertex, the algorithm learns upon querying a vertex that it is equal to the target, or it is given an incident edge that is on the shortest path from the queried vertex to the target. The standard binary search algorithm is simply the case where the graph is a path. Similarly, binary search trees are the case where the edges to the left or right subtrees are given when the queried vertex is unequal to the target. For all undirected, positively weighted graphs, there is an algorithm that finds the target vertex in queries in the worst case.[53]

嘈杂二分查找

[编辑]
In noisy binary search, there is a certain probability that a comparison is incorrect.

Noisy binary search algorithms solve the case where the algorithm cannot reliably compare elements of the array. For each pair of elements, there is a certain probability that the algorithm makes the wrong comparison. Noisy binary search can find the correct position of the target with a given probability that controls the reliability of the yielded position. Every noisy binary search procedure must make at least comparisons on average, where is the binary entropy function and is the probability that the procedure yields the wrong position.[54][55][56] The noisy binary search problem can be considered as a case of the Rényi-Ulam game,[57] a variant of Twenty Questions where the answers may be wrong.[58]

量子二分查找

[编辑]

Classical computers are bounded to the worst case of exactly iterations when performing binary search. Quantum algorithms for binary search are still bounded to a proportion of queries (representing iterations of the classical procedure), but the constant factor is less than one, providing for a lower time complexity on quantum computers. Any exact quantum binary search procedure—that is, a procedure that always yields the correct result—requires at least queries in the worst case, where is the natural logarithm.[59] There is an exact quantum binary search procedure that runs in queries in the worst case.[60] In comparison, Grover's algorithm is the optimal quantum algorithm for searching an unordered list of elements, and it requires queries.[61]

历史

[编辑]

The idea of sorting a list of items to allow for faster searching dates back to antiquity. The earliest known example was the Inakibit-Anu tablet from Babylon dating back to 约200 BCE. The tablet contained about 500 sexagesimal numbers and their reciprocals sorted in lexicographical order, which made searching for a specific entry easier. In addition, several lists of names that were sorted by their first letter were discovered on the Aegean Islands. Catholicon, a Latin dictionary finished in 1286 CE, was the first work to describe rules for sorting words into alphabetical order, as opposed to just the first few letters.[14]

In 1946, John Mauchly made the first mention of binary search as part of the Moore School Lectures, a seminal and foundational college course in computing.[14] In 1957, William Wesley Peterson published the first method for interpolation search.[14][62] Every published binary search algorithm worked only for arrays whose length is one less than a power of two[j] until 1960, when Derrick Henry Lehmer published a binary search algorithm that worked on all arrays.[64] In 1962, Hermann Bottenbruch presented an ALGOL 60 implementation of binary search that placed the comparison for equality at the end, increasing the average number of iterations by one, but reducing to one the number of comparisons per iteration.[13] The uniform binary search was developed by A. K. Chandra of Stanford University in 1971.[14] In 1986, Bernard Chazelle and Leonidas J. Guibas introduced fractional cascading as a method to solve numerous search problems in computational geometry.[51][65][66]

实现问题

[编辑]

尽管二分查找的基本思想相对简单,但其细节却出奇复杂。

乔恩·本特利在为职业程序员开设的一门课程中布置了二分查找的练习,发现90%的学生在数小时后仍未能给出正确的解答。主要原因是算法实现有误而无法运行,或是在极少数边界情况英语Edge case下返回错误答案。[67]1988年发表的一项研究显示,二十本教材中只有五本给出了准确的二分查找代码。[68]此外,本特利自身在1986年出版的《编程珠玑》一书中给出的二分查找实现存在溢出错误,这个错误在二十多年里未被发现。Java编程语言库中的二分查找实现也存在相同的溢出问题,且该问题持续了九年多。[69]

在实际编程中,表示索引的变量通常是固定大小的(整数)。因此在处理非常大的数组时,可能会导致算术溢出。如果使用计算中点,即使的值都在所用数据类型的表示范围内,的值仍可能会超过范围。如果都是非负数,可以通过计算来避免这种情况。[70]

如果循环的退出条件定义不正确,可能会导致无限循环。当超过时,表示搜索失败,必须返回失败的信息。另外,循环应在找到目标元素时退出;若不这么做,那么在循环结束后,必须检查是否成功找到目标元素。本特利发现,大多数在实现二分查找时出错的程序员,都是在定义退出条件时犯了错。[13][71]

库支持

[编辑]

许多编程语言的标准库包含二分查找例程:

  • C语言在其标准库中提供了bsearch()函数,通常使用二分查找实现,尽管官方标准中并未强制要求。[72]
  • C++标准库中提供了binary_search()lower_bound()upper_bound()equal_range()函数。[73]
  • D语言的标准库Phobos在std.range模块中提供了SortedRange类型(由sort()assumeSorted()函数返回),该类型包含contains()equaleRange()lowerBound()trisect()方法,这些方法默认对提供随机访问的范围使用二分查找技术。[74]
  • COBOL提供了SEARCH ALL动词,用于对COBOL有序表执行二分查找。[75]
  • Gosort标准库包包含SearchSearchIntsSearchFloat64sSearchStrings函数,分别实现了通用的二分查找,以及针对整数、浮点数、字符串切片的特定实现。[76]
  • Java在标准java.util包的ArraysCollections类中提供了一组重载binarySearch()静态方法,用于对Java数组和List(列表)执行二分查找。[77][78]
  • Microsoft.NET Framework 2.0在其集合基类中提供了二分查找算法的静态泛型版本,例如System.ArrayBinarySearch<T>(T[] array, T value)方法。[79]
  • 对于Objective-CCocoa框架Mac OS X 10.6及以上版本中提供了NSArray -indexOfObject:inSortedRange:options:usingComparator:方法;[80]苹果的Core Foundation英语Core Foundation C框架也包含CFArrayBSearchValues()函数。[81]
  • Python提供了模块bisect,在插入元素后仍能保持列表的有序状态,而无需每次插入元素后都对列表排序。[82]
  • Ruby的Array类包含一个带有内置近似匹配的bsearch方法。[83]
  • Rust的切片原始类型提供了binary_search()binary_search_by()binary_search_by_key()partition_point()方法。[84]

参见

[编辑]

注释和参考文献

[编辑]

注释

[编辑]
  1. ^ 又称折半查找[4](英語:half-interval search[5],台湾作折半搜尋,大陆又作折半搜索[6],直译为「半区间搜索」)、对数搜索(英語:logarithmic search[7]),英文中又称binary chop[8]chop有「劈、斩」之意)。
  2. ^ The is Big O notation, and is the logarithm. In Big O notation, the base of the logarithm does not matter since every logarithm of a given base is a constant factor of another logarithm of another base. That is, , where is a constant.
  3. ^ Any search algorithm based solely on comparisons can be represented using a binary comparison tree. An internal path is any path from the root to an existing node. Let be the internal path length, the sum of the lengths of all internal paths. If each element is equally likely to be searched, the average case is or simply one plus the average of all the internal path lengths of the tree. This is because internal paths represent the elements that the search algorithm compares to the target. The lengths of these internal paths represent the number of iterations after the root node. Adding the average of these lengths to the one iteration at the root yields the average case. Therefore, to minimize the average number of comparisons, the internal path length must be minimized. It turns out that the tree for binary search minimizes the internal path length. Knuth 1998 proved that the external path length (the path length over all nodes where both children are present for each already-existing node) is minimized when the external nodes (the nodes with no children) lie within two consecutive levels of the tree. This also applies to internal paths as internal path length is linearly related to external path length . For any tree of nodes, . When each subtree has a similar number of nodes, or equivalently the array is divided into halves in each iteration, the external nodes as well as their interior parent nodes lie within two levels. It follows that binary search minimizes the number of average comparisons as its comparison tree has the lowest possible internal path length.[19]
  4. ^ Knuth 1998 showed on his MIX computer model, which Knuth designed as a representation of an ordinary computer, that the average running time of this variation for a successful search is units of time compared to units for regular binary search. The time complexity for this variation grows slightly more slowly, but at the cost of higher initial complexity. [23]
  5. ^ Knuth 1998 performed a formal time performance analysis of both of these search algorithms. On Knuth's MIX computer, which Knuth designed as a representation of an ordinary computer, binary search takes on average units of time for a successful search, while linear search with a sentinel node at the end of the list takes units. Linear search has lower initial complexity because it requires minimal computation, but it quickly outgrows binary search in complexity. On the MIX computer, binary search only outperforms linear search with a sentinel if .[19][28]
  6. ^ Inserting the values in sorted order or in an alternating lowest-highest key pattern will result in a binary search tree that maximizes the average and worst-case search time.[33]
  7. ^ It is possible to search some hash table implementations in guaranteed constant time.[38]
  8. ^ This is because simply setting all of the bits which the hash functions point to for a specific key can affect queries for other keys which have a common hash location for one or more of the functions.[43]
  9. ^ There exist improvements of the Bloom filter which improve on its complexity or support deletion; for example, the cuckoo filter exploits cuckoo hashing to gain these advantages.[43]
  10. ^ That is, arrays of length 1, 3, 7, 15, 31 ...[63]

引用

[编辑]
  1. ^ Sedgewick, Wayne & 谢路云 2012,§3.1.5 有序数组中的二分查找.
  2. ^ 廖永申; 王欣平. 在嵌入式Linux系統上的新一代網路協定轉換軟體開發. 2003年NCS全國計算機會議. 2006-06-12 [2024-07-16] (中文(臺灣)). 
  3. ^ 韦纯福. 基于分治思想的二分搜索技术研究. 大众科技. 2008, (3): 58–59. CNKI DZJI200803022需注册账号查阅 (中文(中国大陆)). 
  4. ^ 骆剑锋. 哈希表与一般查找方法的比较及冲突的解决. 十堰职业技术学院学报. 2007, (5): 96–98. CNKI SYZJ200705034需注册账号查阅 (中文(中国大陆)). 
  5. ^ Williams, Jr., Louis F. A modification to the half-interval search (binary search) method. Proceedings of the 14th ACM Southeast Conference. ACM: 95–101. 1976-04-22. doi:10.1145/503561.503582可免费查阅. 
  6. ^ 王晓东. 一种在TQuery记录集中实现快速搜索的方法. 管理信息系统. 2000, (3): 56–57. CNKI JYXX200003023需注册账号查阅 (中文(中国大陆)). 
  7. ^ 7.0 7.1 Knuth 1998,§6.2.1 ("Searching an ordered table"), subsection "Binary search".
  8. ^ Butterfield & Ngondi 2016,第46頁.
  9. ^ Cormen et al. 2009,第39頁.
  10. ^ 埃里克·韦斯坦因. Binary search. MathWorld. 
  11. ^ 11.0 11.1 Flores, Ivan; Madpis, George. Average binary search length for dense ordered lists. Communications of the ACM. 1971-09-01, 14 (9): 602–603. ISSN 0001-0782. S2CID 43325465. doi:10.1145/362663.362752可免费查阅. 
  12. ^ 12.0 12.1 12.2 Knuth 1998,§6.2.1 ("Searching an ordered table"), subsection "Algorithm B".
  13. ^ 13.0 13.1 13.2 13.3 Bottenbruch, Hermann. Structure and use of ALGOL 60. Journal of the ACM. 1962-04-01, 9 (2): 161–221. ISSN 0004-5411. S2CID 13406983. doi:10.1145/321119.321120可免费查阅.  Procedure is described at p. 214 (§43), titled "Program for Binary Search".
  14. ^ 14.0 14.1 14.2 14.3 14.4 14.5 Knuth 1998,§6.2.1 ("Searching an ordered table"), subsection "History and bibliography".
  15. ^ 15.0 15.1 Kasahara & Morishita 2006,第8–9頁.
  16. ^ 16.0 16.1 16.2 Sedgewick & Wayne 2011,§3.1, subsection "Rank and selection".
  17. ^ 17.0 17.1 17.2 Goldman & Goldman 2008,第461–463頁.
  18. ^ Sedgewick & Wayne 2011,§3.1, subsection "Range queries".
  19. ^ 19.00 19.01 19.02 19.03 19.04 19.05 19.06 19.07 19.08 19.09 19.10 19.11 Knuth 1998,§6.2.1 ("Searching an ordered table"), subsection "Further analysis of binary search".
  20. ^ Knuth 1998,§6.2.1 ("Searching an ordered table"), "Theorem B".
  21. ^ Chang 2003,第169頁.
  22. ^ 22.0 22.1 22.2 Knuth 1997,§2.3.4.5 ("Path length").
  23. ^ 23.0 23.1 Knuth 1998,§6.2.1 ("Searching an ordered table"), subsection "Exercise 23".
  24. ^ Rolfe, Timothy J. Analytic derivation of comparisons in binary search. ACM SIGNUM Newsletter. 1997, 32 (4): 15–19. S2CID 23752485. doi:10.1145/289251.289255可免费查阅. 
  25. ^ 25.0 25.1 Khuong, Paul-Virak; Morin, Pat. Array Layouts for Comparison-Based Searching. Journal of Experimental Algorithmics. 2017, 22. Article 1.3. S2CID 23752485. arXiv:1509.05053可免费查阅. doi:10.1145/3053370. 
  26. ^ Knuth 1997,§2.2.2 ("Sequential Allocation").
  27. ^ 27.0 27.1 27.2 27.3 Beame, Paul; Fich, Faith E. Optimal bounds for the predecessor problem and related problems. Journal of Computer and System Sciences. 2001, 65 (1): 38–72. doi:10.1006/jcss.2002.1822可免费查阅. 
  28. ^ Knuth 1998,Answers to Exercises (§6.2.1) for "Exercise 5".
  29. ^ Knuth 1998,§6.2.1 ("Searching an ordered table").
  30. ^ Knuth 1998,§5.3.1 ("Minimum-Comparison sorting").
  31. ^ Sedgewick & Wayne 2011,§3.2 ("Ordered symbol tables").
  32. ^ Sedgewick & Wayne 2011,§3.2 ("Binary Search Trees"), subsection "Order-based methods and deletion".
  33. ^ Knuth 1998,§6.2.2 ("Binary tree searching"), subsection "But what about the worst case?".
  34. ^ Sedgewick & Wayne 2011,§3.5 ("Applications"), "Which symbol-table implementation should I use?".
  35. ^ Knuth 1998,§5.4.9 ("Disks and Drums").
  36. ^ Knuth 1998,§6.2.4 ("Multiway trees").
  37. ^ Knuth 1998,§6.4 ("Hashing").
  38. ^ Knuth 1998,§6.4 ("Hashing"), subsection "History".
  39. ^ Dietzfelbinger, Martin; Karlin, Anna; Mehlhorn, Kurt; Meyer auf der Heide, Friedhelm; Rohnert, Hans; Tarjan, Robert E. Dynamic perfect hashing: upper and lower bounds. SIAM Journal on Computing. August 1994, 23 (4): 738–761. doi:10.1137/S0097539791194094. 
  40. ^ Morin, Pat. Hash tables (PDF): 1. [2016-03-28]. (原始内容存档 (PDF)于2022-10-09). 
  41. ^ Knuth 2011,§7.1.3 ("Bitwise Tricks and Techniques").
  42. ^ 42.0 42.1 Silverstein, Alan, Judy IV shop manual (PDF), Hewlett-Packard: 80–81, (原始内容存档 (PDF)于2022-10-09) 
  43. ^ 43.0 43.1 Fan, Bin; Andersen, Dave G.; Kaminsky, Michael; Mitzenmacher, Michael D. Cuckoo filter: practically better than Bloom. Proceedings of the 10th ACM International on Conference on Emerging Networking Experiments and Technologies: 75–88. 2014. doi:10.1145/2674005.2674994可免费查阅. 
  44. ^ Bloom, Burton H. Space/time trade-offs in hash coding with allowable errors. Communications of the ACM. 1970, 13 (7): 422–426. CiteSeerX 10.1.1.641.9096可免费查阅. S2CID 7931252. doi:10.1145/362686.362692. 
  45. ^ Knuth 1998,§6.2.1 ("Searching an ordered table"), subsection "An important variation".
  46. ^ Knuth 1998,§6.2.1 ("Searching an ordered table"), subsection "Algorithm U".
  47. ^ Moffat & Turpin 2002,第33頁.
  48. ^ 48.0 48.1 48.2 Knuth 1998,§6.2.1 ("Searching an ordered table"), subsection "Interpolation search".
  49. ^ Knuth 1998,§6.2.1 ("Searching an ordered table"), subsection "Exercise 22".
  50. ^ Perl, Yehoshua; Itai, Alon; Avni, Haim. Interpolation search—a log log n search. Communications of the ACM. 1978, 21 (7): 550–553. S2CID 11089655. doi:10.1145/359545.359557可免费查阅. 
  51. ^ 51.0 51.1 51.2 Chazelle, Bernard; Liu, Ding. Lower bounds for intersection searching and fractional cascading in higher dimension. 33rd ACM Symposium on Theory of Computing. ACM: 322–329. 2001-07-06 [2018-06-30]. ISBN 978-1-58113-349-3. doi:10.1145/380752.380818. 
  52. ^ Chazelle, Bernard; Liu, Ding. Lower bounds for intersection searching and fractional cascading in higher dimension (PDF). Journal of Computer and System Sciences. 2004-03-01, 68 (2): 269–284 [2018-06-30]. CiteSeerX 10.1.1.298.7772可免费查阅. ISSN 0022-0000. doi:10.1016/j.jcss.2003.07.003. (原始内容存档 (PDF)于2022-10-09) (英语). 
  53. ^ Emamjomeh-Zadeh, Ehsan; Kempe, David; Singhal, Vikrant. Deterministic and probabilistic binary search in graphs. 48th ACM Symposium on Theory of Computing: 519–532. 2016. arXiv:1503.00805可免费查阅. doi:10.1145/2897518.2897656. 
  54. ^ Ben-Or, Michael; Hassidim, Avinatan. The Bayesian learner is optimal for noisy binary search (and pretty good for quantum as well) (PDF). 49th Symposium on Foundations of Computer Science: 221–230. 2008. ISBN 978-0-7695-3436-7. doi:10.1109/FOCS.2008.58. (原始内容存档 (PDF)于2022-10-09). 
  55. ^ Pelc, Andrzej. Searching with known error probability. Theoretical Computer Science. 1989, 63 (2): 185–202. doi:10.1016/0304-3975(89)90077-7可免费查阅. 
  56. ^ Rivest, Ronald L.; Meyer, Albert R.; Kleitman, Daniel J.; Winklmann, K. Coping with errors in binary search procedures. 10th ACM Symposium on Theory of Computing. doi:10.1145/800133.804351可免费查阅. 
  57. ^ Pelc, Andrzej. Searching games with errors—fifty years of coping with liars. Theoretical Computer Science. 2002, 270 (1–2): 71–109. doi:10.1016/S0304-3975(01)00303-6可免费查阅. 
  58. ^ Rényi, Alfréd. On a problem in information theory. Magyar Tudományos Akadémia Matematikai Kutató Intézetének Közleményei. 1961, 6: 505–516. MR 0143666 (匈牙利语). 
  59. ^ Høyer, Peter; Neerbek, Jan; Shi, Yaoyun. Quantum complexities of ordered searching, sorting, and element distinctness. Algorithmica. 2002, 34 (4): 429–448. S2CID 13717616. arXiv:quant-ph/0102078可免费查阅. doi:10.1007/s00453-002-0976-3. 
  60. ^ Childs, Andrew M.; Landahl, Andrew J.; Parrilo, Pablo A. Quantum algorithms for the ordered search problem via semidefinite programming. Physical Review A. 2007, 75 (3). 032335. Bibcode:2007PhRvA..75c2335C. S2CID 41539957. arXiv:quant-ph/0608161可免费查阅. doi:10.1103/PhysRevA.75.032335. 
  61. ^ Grover, Lov K. A fast quantum mechanical algorithm for database search. 28th ACM Symposium on Theory of Computing. Philadelphia, PA: 212–219. 1996. arXiv:quant-ph/9605043可免费查阅. doi:10.1145/237814.237866. 
  62. ^ Peterson, William Wesley. Addressing for random-access storage. IBM Journal of Research and Development. 1957, 1 (2): 130–146. doi:10.1147/rd.12.0130. 
  63. ^ "2n−1". OEIS A000225 互联网档案馆存檔,存档日期2016-06-08.. Retrieved 7 May 2016.
  64. ^ Lehmer, Derrick. Teaching combinatorial tricks to a computer. Proceedings of Symposia in Applied Mathematics. 1960, 10: 180–181. ISBN 9780821813102. doi:10.1090/psapm/010可免费查阅. 
  65. ^ Chazelle, Bernard; Guibas, Leonidas J. Fractional cascading: I. A data structuring technique (PDF). Algorithmica. 1986, 1 (1–4): 133–162. CiteSeerX 10.1.1.117.8349可免费查阅. S2CID 12745042. doi:10.1007/BF01840440. 
  66. ^ Chazelle, Bernard; Guibas, Leonidas J., Fractional cascading: II. Applications (PDF), Algorithmica, 1986, 1 (1–4): 163–191, S2CID 11232235, doi:10.1007/BF01840441 
  67. ^ Bentley 2000,§4.1 ("The Challenge of Binary Search").
  68. ^ Pattis, Richard E. Textbook errors in binary searching. SIGCSE Bulletin. 1988, 20: 190–194. doi:10.1145/52965.53012. 
  69. ^ Bloch, Joshua. Extra, extra – read all about it: nearly all binary searches and mergesorts are broken. Google Research Blog. 2006-06-02 [2016-04-21]. (原始内容存档于2016-04-01). 
  70. ^ Ruggieri, Salvatore. On computing the semi-sum of two integers (PDF). Information Processing Letters. 2003, 87 (2): 67–71 [2016-03-19]. CiteSeerX 10.1.1.13.5631可免费查阅. doi:10.1016/S0020-0190(03)00263-1. (原始内容存档 (PDF)于2006-07-03). 
  71. ^ Bentley 2000,§4.4 ("Principles").
  72. ^ bsearch – binary search a sorted table. The Open Group Base Specifications 7th. The Open Group. 2013 [2016-03-28]. (原始内容存档于2016-03-21). 
  73. ^ Stroustrup 2013,第945頁.
  74. ^ std.range - D Programming Language. dlang.org. [2020-04-29]. 
  75. ^ Unisys, COBOL ANSI-85 programming reference manual 1: 598–601, 2012 
  76. ^ Package sort. The Go Programming Language. [2016-04-28]. (原始内容存档于2016-04-25). 
  77. ^ java.util.Arrays. Java Platform Standard Edition 8 Documentation. Oracle Corporation. [2016-05-01]. (原始内容存档于2016-04-29). 
  78. ^ java.util.Collections. Java Platform Standard Edition 8 Documentation. Oracle Corporation. [2016-05-01]. (原始内容存档于2016-04-23). 
  79. ^ List<T>.BinarySearch method (T). Microsoft Developer Network. [2016-04-10]. (原始内容存档于2016-05-07). 
  80. ^ NSArray. Mac Developer Library. Apple Inc. [2016-05-01]. (原始内容存档于2016-04-17). 
  81. ^ CFArray. Mac Developer Library. Apple Inc. [2016-05-01]. (原始内容存档于2016-04-20). 
  82. ^ 8.6. bisect — Array bisection algorithm. The Python Standard Library. Python Software Foundation. [2018-03-26]. (原始内容存档于2018-03-25). 
  83. ^ Fitzgerald 2015,第152頁.
  84. ^ Primitive Type slice. The Rust Standard Library. The Rust Foundation. 2024 [2024-05-25]. 

来源

[编辑]


外部链接

[编辑]