【leetcode】719. Find K-th Smallest Pair Distance

题目如下:

解题思路:对于这一类知道上限和下限,求第N位是什么的题目,可以先看看二分查找的方法可不可行。首先对nums进行排序,很显然任意两个元素距离绝对值最小是0,最大是nums[-1] - nums[0],所以第N小的距离肯定在 0 ~ (nums[-1] - nums[0]) 之间。采用二分查找的话,题目就变成了输入一个数值n,判断是不是第N小的距离。怎么判断n是否是第N小的距离呢?因为nums是有序的,对于nums[i]来说,只要找到(n + nums[i]) 在nums中所处的位置,设为j,那么nums[i] 与下标在[i+1,j]之间的元素的距离均小于n,与在[j+1,len(nums)-1]之间元素距离均大于n,求下标j同样可以采用二分查找。看起来很完美了,但是还有关键的一个地方,(n + nums[i])  在nums中不一定存在,或者即使存在但是会有多个值。因此需要找到(n + nums[i]) 在nums中所处的左右两个位置的下标,如果两个下标不一样,则表示不存在;下标的差值表示存在多少个(n + nums[i]) 。

代码如下:

class Solution(object):
    def smallestDistancePair(self, nums, k):
        import bisect
        nums.sort()
        low,high = 0, nums[-1] - nums[0]
        while low <= high:
            mid = (low + high) // 2
            less ,equal = 0,0
            for i,v in enumerate(nums):
                left = (bisect.bisect_left(nums,v+mid) - 1 -i)
                less += left
                right = bisect.bisect_right(nums,v+mid) - 1 -i
                equal += (right - left)
            if less >= k:
                high = mid - 1
            elif less + equal < k:
                low = mid + 1
            elif less == k and equal == 0:
                high = mid - 1
            else:
                break
        return mid

 

posted @ 2018-09-07 13:17  seyjs  阅读(253)  评论(0编辑  收藏  举报