Fork me on GitHub

【python刷题】二分查找

二分查找模板

def binarySearch(nums, target):
    left = 0
    right = len(nums) - 1
    while left <= right:
        mid = (right + left) // 2
        if nums[mid] == target:
            return True
        elif nums[mid] < target:
            left = mid + 1
        else:
            right = mid - 1
    return False

nums = [1,2,3,4,5,7]
res = binarySearch(nums, 7)
print(res)

寻找左侧边界的二分查找

def binarySearch(nums, target):
    left = 0
    right = len(nums)
    while left < right:
        mid = (right + left) // 2
        if nums[mid] == target:
            right = mid
        elif nums[mid] < target:
            left = mid + 1
        else:
            right = mid
    if left == len(nums):
        return -1
    if left == 0 and nums[left] != target:
        return -1
    return left

nums = [1,1,2,2,2,3]
res = binarySearch(nums, 2)
print(res)

寻找右侧边界的二分查找

def binarySearch(nums, target):
    left = 0
    right = len(nums)
    while left < right:
        mid = (right + left) // 2
        if nums[mid] == target:
            left = mid + 1
        elif nums[mid] < target:
            left = mid + 1
        else:
            right = mid
    if right >= len(nums) and nums[right - 1] != target:
        return -1
    return right - 1

nums = [1,2,2,2,2,3,3,4]
res = binarySearch(nums, -2)
print(res)

当数组中存在重复的元素的时候,我们要返回左右边界的时候,当查找到该元素时,我们不能返回True或者Fasle,而是要不断的缩小边界。

包裹运输问题

def shipWithinDays(weights, d):
    left = max(weights)
    right = sum(weights) + 1 # 船的载重区间[left,right)
    while left < right:
        mid = (right + left) // 2
        if canFinish(weights, d, mid):
             right = mid
        else:
            left = mid + 1
    return left

def canFinish(w, d, cap):
    i = 0
    for day in range(d):
        maxCap = cap
        while maxCap - w[i] >= 0:
            i += 1
            if i == len(w):
                return True
            maxCap = maxCap - w[i]
    return False

该题代码还有点问题,没有通过leetcode上的全部用例,掌握思想就好。
最后,首先思考使用 for 循环暴力解决问题,观察代码是否如下形式:

for i in range(n):
    if isOK(i):
        return answer

如果是,那么就可以使用二分搜索优化搜索空间:如果要求最小值就是搜索左侧边界的二分,如果要求最大值就用搜索右侧边界的二分。

labuladong的算法小抄

posted @ 2021-02-03 16:14  西西嘛呦  阅读(126)  评论(0编辑  收藏  举报