用bisect模块解答二分算法题
1.bisect模块介绍
bisect模块包括两个主要函数 bisect和insort
这两个函数都用二分查找算法来在有序序列中查找或者插入元素,在插入这个元素后,原来的序列还能保持升序
bisect/ bisect_left / bisect_right 返回该元素插入的index
insort/ insort_left / insort_right 把这个元素插入到序列里
left和right,是指当序列里存在同样的元素时,新元素插入的位置在它的左边还是右边
2.利用bisect模块解决二分算法题 (算法题来自于leetcode)
例1:
给定一个排序数组和一个目标值,在数组中找到目标值,并返回其索引。如果目标值不存在于数组中,返回它将会被按顺序插入的位置。你可以假设数组中无重复元素。
示例 1:
输入: [1,3,5,6], 5
输出: 2
示例 2:
输入: [1,3,5,6], 2
输出: 1
示例 3:
输入: [1,3,5,6], 7
输出: 4
示例 4:
输入: [1,3,5,6], 0
输出: 0
class Solution(object): def searchInsert(self, nums, target): """ :type nums: List[int] :type target: int :rtype: int """ return bisect.bisect_left(nums,target)
例2:
你是产品经理,目前正在带领一个团队开发新的产品。不幸的是,你的产品的最新版本没有通过质量检测。由于每个版本都是基于之前的版本开发的,所以错误的版本之后的所有版本都是错的。
假设你有 n
个版本 [1, 2, ..., n]
,你想找出导致之后所有版本出错的第一个错误的版本。
你可以通过调用 bool isBadVersion(version)
接口来判断版本号 version
是否在单元测试中出错。实现一个函数来查找第一个错误的版本。你应该尽量减少对调用 API 的次数。
示例:给定 n = 5,并且 version = 4 是第一个错误的版本。
调用 isBadVersion(3) -> false
调用 isBadVersion(5) -> true
调用 isBadVersion(4) -> true
所以,4 是第一个错误的版本。
大牛的答案
# The isBadVersion API is already defined for you. # @param version, an integer # @return a bool # def isBadVersion(version): class Solution(object): def firstBadVersion(self, n): """ :type n: int :rtype: int """ "from other person, impressively!!!" class Wrap: def __getitem__(self, i): return isBadVersion(i) return bisect.bisect(Wrap(), False, 0, n)
我的答案
# The isBadVersion API is already defined for you. # @param version, an integer # @return a bool # def isBadVersion(version): class Solution(object): def firstBadVersion(self, n): """ :type n: int :rtype: int """ start = 1 end = n mid = (start + end ) // 2 while (start < end): if isBadVersion(mid) == False: start = mid + 1 else: end = mid mid = (start + end ) // 2 return mid
这个答案来自于leetcode上的大牛,看到这个题目的时候,第一反映,也是想用bisect,但是没有任何idea;所以老老实实手写了一个二分查找..
Wrap实乃精妙之作,满心佩服!
posted on 2018-11-16 14:58 cherryjing0629 阅读(170) 评论(0) 编辑 收藏 举报