leetcode 004-006

这几天疯狂考试,具体的代码思路就不分析了,第四第五题都用自己的方法实现了,因为是暴力刚,被卡边界的过程挺惨的。说起来,这就是有思路和没思路的区别了。

更新:考完试,想起来还欠着这个。把之前没注意的都补上

leetcode 004

代码实现

class Solution:
    
    def delBothArrays(self, lst_nums, del_len):
        """
        :type lst_nums: list[list[int]]
        :type del_len: int
        :rtype: list[int]
        """
        nums1, nums2 = lst_nums
        for t in range(del_len):
            if len(nums1) == 0:
                nums2.pop()
            elif len(nums2) == 0:
                nums1.pop()
            else:
                if nums1[-1] >= nums2[-1]:
                    nums1.pop()
                else:
                    nums2.pop()
    
    def singleLstMedian(self, nums):
        """
        :type nums: List[int]
        :rtype: float
        """
        if len(nums) % 2:
            i = (len(nums)-1) // 2
            return nums[i]
        else:
            i = (len(nums)-2) // 2
            return (nums[i] + nums[i+1]) / 2
    
    def findMedianSortedArrays(self, nums1, nums2):
        """
        :type nums1: List[int]
        :type nums2: List[int]
        :rtype: float
        """
        i = len(nums1) - 1
        j = len(nums2) - 1
        if i == -1:
            return self.singleLstMedian(nums2)
        elif j == -1:
            return self.singleLstMedian(nums1)
        if (i + j) % 2:
            del_len = (i + j + 1) // 2
            self.delBothArrays([nums1,nums2], del_len)
            if len(nums1) == 0:
                median = nums2[-1]
            elif len(nums2) == 0:
                median = nums1[-1]
            else:
                median = nums1[-1] if nums1[-1] >= nums2[-1] else nums2[-1]
        else:
            del_len = (i + j) // 2
            self.delBothArrays([nums1,nums2], del_len)
            if len(nums1) == 1:
                lst_new = nums1[-1:] + nums2[-2:]
            elif len(nums2) == 1:
                lst_new = nums1[-2:] + nums2[-1:]
            else:
                lst_new = nums1[-2:] + nums2[-2:]
            lst_new.sort()
            median = (lst_new[-1] + lst_new[-2]) / 2
        return median

上面是我自己的思路,简单来说就是将两个数组共同从后向前删去一半(或减一的一半)的数。需要注意的是,我这两个数组并没有合并。删完之后的两个数组,就会暴露出两个疑似中位数的值,最后比较出来就可以了。时间复杂度是O(m+n)。

而解答里的思路,怎么说非常的巧妙。具体的思路如下:

只需要保证 B[j-1]<A[i] 和 A[i-1]<B[j] ,就有max(left) < min(right)。同时 i 的检索使用了二分技巧,而 j 由 len(left) = len(right) 可以找到一个计算式:\(j = \frac{}m+n+1{2}-1\)。需要注意的是,+1是非常重要的,这在总数为奇的情况下,会保证左边那堆数永远比右边的多一个。

解答代码如下:

class Solution:
    def findMedianSortedArrays(self, nums1, nums2):
        """
        :type nums1: List[int]
        :type nums2: List[int]
        :rtype: float
        """
        m, n = len(nums1), len(nums2)
        if m > n:
            m, n, nums1, nums2 = n, m, nums2, nums1
        imin, imax, half_len = 0, m, (m + n + 1) // 2
        while imin <= imax:
            i = (imin + imax) // 2
            j = half_len - i
            if i > 0 and nums1[i-1] > nums2[j]:
                imax = i - 1
            elif i < m and nums2[j-1] > nums1[i]:
                imin = i + 1
            else:
                if i == 0: max_left = nums2[j-1]
                elif j == 0: max_left = nums1[i-1]
                else: max_left = max(nums1[i-1], nums2[j-1])
                
                if (m + n) % 2 == 1:
                    return max_left
                
                if i == m: min_right = nums2[j]
                elif j == n: min_right = nums1[i]
                else: min_right = min(nums1[i], nums2[j])
                    
                return (max_left + min_right) / 2

leetcode 005

代码实现

class Solution:
    def longestPalindrome(self, s):
        """
        :type s: str
        :rtype: str
        """
        if len(s) == 0 or len(s) == 1:
            return s
        lst_init = list(s)
        lst_cmp = []
        length = 0
        pos = 1
        is_odd = True
        pop_num = lst_init.pop()
        for i in range(len(s) - 1, 0, -1):
            lst_cmp.append(pop_num)
            offset = 1
            while True:     # 找出偶回文串
                if len(lst_cmp) < offset or len(lst_init) < offset:
                    break
                if lst_cmp[-offset] == lst_init[-offset]:
                    if offset > length:
                        length = offset
                        pos = i
                        is_odd = False
                    offset += 1
                else:
                    break
            pop_num = lst_init.pop()
            offset = 1
            while True:     # 找出奇回文串
                if len(lst_cmp) < offset or len(lst_init) < offset:
                    break
                if lst_cmp[-offset] == lst_init[-offset]:
                    if offset >= length: # 奇大于偶,但会导致不稳定
                        length = offset
                        pos = i
                        is_odd = True
                    offset += 1
                else:
                    break
        if not is_odd:
            right = pos + length
            left = pos - length
        else:
            if length == 0:
                return s[-1]  # 如果不规则,返回最后一个
            right = pos + length
            left = pos - 1 - length
        return s[left:right]

这道题我的解题思路其实是挺清晰的,简而言之就是用两个队列并排比较。代码看起来很复杂,其实时间复杂度为O(n^2),空间复杂度为O(n),算是还好。做的时候确实被很多边界卡得很厉害,等以后看看有没有办法简化。

看了下答案,竟然有三四种解法,之后一个个去琢磨。

leetcode 006

代码实现

这道题没有做出来。

class Solution:
    def convert(self, s, numRows):
        """
        :type s: str
        :type numRows: int
        :rtype: str
        """
        if numRows == 1 or numRows >= len(s):
            return s

        L = [''] * numRows
        index, step = 0, 1

        for x in s:
            L[index] += x
            if index == 0:
                step = 1
            elif index == numRows -1:
                step = -1
            index += step

        return ''.join(L)
posted @ 2019-01-06 21:21  Wunsam_Chan  阅读(190)  评论(0编辑  收藏  举报