剑指offer【50~59】

题目链接:

剑指offer 50-59


目录:

50. 第一个只出现一次的字符位置
51. 数组中的逆序对
52. 两个链表的第一个公共结点
53. 数字在排序数组中出现的次数
54. 二叉查找树的第 K 个结点
55.1 二叉树的深度
55.2 平衡二叉树
56. 数组中只出现一次的数字
57.1 和为 S 的两个数字
57.2 和为 S 的连续正数序列
58.1 翻转单词顺序列
58.2 左旋转字符串
59. 滑动窗口的最大值


Python 实现:

50. 第一个只出现一次的字符位置

使用大小为 256 的数组记录每个字符出现的次数。遍历两遍即可。

# -*- coding:utf-8 -*-
class Solution:
    def FirstNotRepeatingChar(self, s):
        # write code here
        dic = [0] * 256
        for i in range(len(s)):
            dic[ord(s[i])] += 1
        for i in range(len(s)):
            if dic[ord(s[i])] == 1:
                return i
        return -1

51. 数组中的逆序对
  • 常规方法 O(n^2),先 pass。可以使用归并排序的思想,在归并的过程中统计逆序对的数量。
  • 比如在归并过程中,left = [1,3,5,7,9],right = [2,4,6,8,10],发现 3 > 2,则 left[1:] 的所有数都比 right[0] 大,就累加逆序对数量,后面的也同理。
  • 详情可以参考博客 剑指Offer(三十五):数组中的逆序对
  • 这样时间复杂度为 O(nlogn),空间复杂度为 O(n)。但是 Python 卡了一下时间,AC 了 75%。
# -*- coding:utf-8 -*-
class Solution:
    def InversePairs(self, data):
        # write code here
        self.ans = 0  # 逆序对数量
        self.mergeSort(data)   # 归并排序
        return self.ans % 1000000007
    
    def mergeSort(self, nums):
        # 递归过程
        if len(nums) <= 1:
            return nums
        mid = len(nums) // 2
        left = self.mergeSort(nums[:mid])
        right = self.mergeSort(nums[mid:])
        return self.merge(left, right)

    # 归并过程 + 统计逆序对数量
    def merge(self, left, right):
        result = []  # 保存归并后的结果
        i = j = 0
        left_len = len(left)
        while i < len(left) and j < len(right):
            if left[i] <= right[j]:
                result.append(left[i])
                i += 1
            else:  # left[i] > right[j]
                self.ans += (left_len - i)  # 核心:说明 nums[i:] 都大于 nums[j]
                result.append(right[j])
                j += 1
        result = result + left[i:] + right[j:] # 剩余的元素直接添加到末尾
        return result

52. 两个链表的第一个公共结点
  • 设 A 的长度为 a + c,B 的长度为 b + c,其中 c 为尾部公共部分长度,可知 a + c + b = b + c + a。
  • 当访问链表 A 的指针 L1 访问到链表尾部时,令它从链表 B 的头部重新开始访问链表 B;同样地,当访问链表 B 的指针 L2 访问到链表尾部时,令它从链表 A 的头部重新开始访问链表 A。这样就能控制 L1 和 L2 能同时访问到公共结点。即 L1 总共走了 (a + c) + b,L2 总共走了 (b + c) + a。
  • 这样,链表 A 和链表 B 都走了 a + b + c 步,时间复杂度为 O(n),空间复杂度为 O(1)。
# -*- coding:utf-8 -*-
# class ListNode:
#     def __init__(self, x):
#         self.val = x
#         self.next = None
class Solution:
    def FindFirstCommonNode(self, pHead1, pHead2):
        # write code here
        l1, l2 = pHead1, pHead2
        while l1 != l2:
            l1 = l1.next if l1 != None else pHead2
            l2 = l2.next if l2 != None else pHead1
        return l1 # 或者 l2

53. 数字在排序数组中出现的次数

排序数组,很明显二分查找,找到第一个 >= k 的元素索引以及第一个 > k 的元素索引,两者相减即为答案,即 lowerBound - upperBound。时间复杂度为 O(logn),空间复杂度为 O(1)。

# -*- coding:utf-8 -*-
class Solution:
    def GetNumberOfK(self, data, k):
        # write code here
        if not data:
            return 0
        lo, hi = 0, len(data) - 1
        while lo < hi:
            mid = lo + (hi - lo) // 2
            if data[mid] >= k:
                hi = mid
            elif data[mid] < k:
                lo = mid + 1
        ind1 = lo if data[lo] >= k else len(data)  # >=k 的第一个元素
        lo, hi = 0, len(data) - 1
        while lo < hi:
            mid = lo + (hi - lo) // 2
            if data[mid] > k:
                hi = mid
            elif data[mid] <= k:
                lo = mid + 1
        ind2 = lo if data[lo] > k else len(data)  # # >k 的第一个元素
        return ind2 - ind1  # 相减即为答案

更简洁的,可以使用 Python 的 bisect 模块中的函数实现。可以参考博客:二分查找及其变形与Python的bisect模块的关系。

# -*- coding:utf-8 -*-
import bisect
class Solution:
    def GetNumberOfK(self, data, k):
        # write code here
        return bisect.bisect_right(data, k) - bisect.bisect_left(data, k)

54. 二叉查找树的第 K 个结点

中序遍历,找到第 k 个数即可。

#     def __init__(self, x):
#         self.val = x
#         self.left = None
#         self.right = None
class Solution:
    # 返回对应节点TreeNode
    def __init__(self):
        self.cnt = 0
        
    def KthNode(self, pRoot, k):
        # write code here
        if not pRoot or self.cnt >= k:  # self.cnt >= k 剪枝
            return None
        left = self.KthNode(pRoot.left, k)
        if left:
            return left
        self.cnt += 1
        if self.cnt == k:
            return pRoot
        right = self.KthNode(pRoot.right, k)
        if right:
            return right
        return None

55.1 二叉树的深度

递归左右子树找最大高度。注意:根的高度为 1。

# -*- coding:utf-8 -*-
# class TreeNode:
#     def __init__(self, x):
#         self.val = x
#         self.left = None
#         self.right = None
class Solution:
    def TreeDepth(self, pRoot):
        # write code here
        if not pRoot:
            return 0
        return max(self.TreeDepth(pRoot.left), self.TreeDepth(pRoot.right)) + 1
55.2 平衡二叉树

平衡二叉树(AVL)的左右子树的高度差不超过 1,因此引入求高度的函数。并且,判断该树的每个结点的左右子树是否也满足 AVL 的定义。

# -*- coding:utf-8 -*-
# class TreeNode:
#     def __init__(self, x):
#         self.val = x
#         self.left = None
#         self.right = None
class Solution:
    def IsBalanced_Solution(self, pRoot):
        # write code here
        if not pRoot:
            return True
        if abs(self.TreeDepth(pRoot.left) - self.TreeDepth(pRoot.right)) > 1:  # 不满足 AVL 的定义
            return False
        return self.IsBalanced_Solution(pRoot.left) and self.IsBalanced_Solution(pRoot.right)
        
    def TreeDepth(self, pRoot):
        if not pRoot:
            return 0
        return max(self.TreeDepth(pRoot.left), self.TreeDepth(pRoot.right)) + 1

改进:自顶向下在调用求树的高度的函数时,有很多重复的操作。因此,可以使用自底向上的方法,一边计算树的高度,一边判断是否是 AVL 树。

# -*- coding:utf-8 -*-
# class TreeNode:
#     def __init__(self, x):
#         self.val = x
#         self.left = None
#         self.right = None
class Solution:
    def IsBalanced_Solution(self, pRoot):
        # write code here
        self.isBalance = True
        self.TreeDepth(pRoot)  # 在求树的高度过程中判断 AVL
        return self.isBalance
        
    def TreeDepth(self, pRoot):
        if not pRoot:
            return 0
        left = self.TreeDepth(pRoot.left)
        right = self.TreeDepth(pRoot.right)
        if abs(left - right) > 1:   # 不满足 AVL 定义,修改 self.isBalance 的值
            self.isBalance = False
        return max(left, right) + 1

56. 数组中只出现一次的数字

如果只出现一次的数字只有一个,很好做,就是全部异或即可。但是,只出现一次的数字有两个怎么做呢?

  • 假设只出现一次的数字为 x 和 y,首先,还是先全部异或得到一个结果 xor,则 x ^ y = xor(相同的数字异或后抵消为 0)
  • 因为 x 和 y 肯定不同,那么它们的二进制表示中肯定有一位一个是 0, 一个是 1。比如 x = 6 (110),y = 4 (100),xor = 2 (10),则我们对 xor 从后往前找到倒数第一个 1 的位置 bits(倒数第 2 位),则以这个 1 为界限,x 和 y 的倒数第 2 位肯定是不同。
  • 因此,对原来的数组重新异或,将 bits 位置为 1 的全部异或,bits 为 0 的全部异或,就是最终的两个返回的结果。
# -*- coding:utf-8 -*-
class Solution:
    # 返回[a,b] 其中ab是出现一次的两个数字
    def FindNumsAppearOnce(self, array):
        # write code here
        xor = 0
        ans = [0, 0]
        for num in array:
            xor ^= num
        bits = 0  # 找到倒数第 i 位为 1
        while xor & 1 << bits == 0:
            bits += 1
        for num in array:
            if num >> bits & 1 == 1:  # num 的倒数第 i 位为 1
                ans[0] ^= num
            else:  # num 的倒数第 i 位为 0
                ans[1] ^= num
        return ans

57.1 和为 S 的两个数字

双指针指向首尾,往中间走,碰到第一对和为 S 的就是答案。

# -*- coding:utf-8 -*-
class Solution:
    def FindNumbersWithSum(self, array, tsum):
        # write code here
        lo, hi = 0, len(array) - 1
        while lo < hi:   # 不能指向相同的数字
            if array[lo] + array[hi] == tsum:
                return [array[lo], array[hi]]
            elif array[lo] + array[hi] < tsum:
                lo += 1
            else:
                hi -= 1
        return []
57.2 和为 S 的连续正数序列

经典的滑动窗口例题,只不过该题的窗口大小不确定。用一个遍历 left 记录窗口左侧的值,window_sum 将窗口中数进行累加。当发现 window_sum >= S 时,相等时输出结果,更新 left 和 window_sum。

# -*- coding:utf-8 -*-
class Solution:
    def FindContinuousSequence(self, tsum):
        # write code here
        if tsum <= 1:  # 特殊情况
            return []
        ans = []
        left, window_sum = 1, 0  # left 记录窗口左界
        for i in range(1, (tsum + 1) // 2 + 1):  # 窗口中至少两个数
            window_sum += i  # 窗口中的累加和
            while window_sum >= tsum:
                if window_sum == tsum:  # 输出一组结果
                    ans.append([j for j in range(left, i + 1)])
                window_sum -= left  # 修改窗口中的累加和
                left += 1  # 修改窗口的左界
        return ans

58.1 翻转单词顺序列

如果要求空间复杂度为 O(1),即只能使用字符串本身,该怎么操作呢?

  • 如 s = "I am a student.",先将各个单词翻转,得 s = ".tneduts a ma I";
  • 再将整个字符串翻转,得 s = "student. a am I"

这样就可以在字符串本身上做修改,使得空间复杂度为 O(1) 了。

注意:但是使用 Python 实现得话,由于不能修改字符串本身,所以还是先要将字符串转化为列表。但是如果使用 C++ 的字符数组,就不用开辟空间了。

# -*- coding:utf-8 -*-
class Solution:
    def ReverseSentence(self, s):
        # write code here
        def reverseWord(s, i, j):
            while i < j:
                s[i], s[j] = s[j], s[i]
                i += 1
                j -= 1
        
        s = list(s)  # python 中 s 不能修改,先转化为 list 
        start = 0  # start 指向当前单词的起始位置
        lens = len(s)
        for i in range(lens + 1):
            if i == lens or s[i] == ' ':
                reverseWord(s, start, i - 1) # 先翻转每个单词
                start = i + 1  # 下一个单词的起始位置
        reverseWord(s, 0, lens - 1)  # 再将整个句子翻转
        return "".join(s)
58.2 左旋转字符串

如果也不能使用空间,怎么做呢?参考上面的 58.1,如 s = "abcXYZdef",n = 3,先将 "abc" 和 "XYZdef" 分别翻转,得到 "cbafedZYX",然后再把整个字符串翻转得到 "XYZdefabc"。这样就可以空间复杂度为 O(1) 了。

# -*- coding:utf-8 -*-
class Solution:
    def LeftRotateString(self, s, n):
        # write code here
        def reverseWord(s, i, j):
            while i < j:
                s[i], s[j] = s[j], s[i]
                i += 1
                j -= 1
        
        if not s:
            return ""
        s = list(s)
        lens = len(s)
        n %= lens  # 循环左移
        reverseWord(s, 0, n - 1)  # 先将前 n 个字符翻转
        reverseWord(s, n, lens - 1)  # 再将后面的字符翻转
        reverseWord(s, 0, lens - 1)  # 最后将整个字符串翻转
        return "".join(s)  # 再转化回字符串

59. 滑动窗口的最大值

使用双向递减队列,队列中始终维护的是窗口中的递减值。

  • 如果队列的大小达到了 size,则应该把队列最前面的数字删除掉。
  • 如果从最右边加入了一个较大的数字,需要从右开始退队列(while 循环),使得队列是单调递减的。
  • 由于双向队列中,从左边出队列和从右边出队列的操作时间复杂度均为 O(1),因此该算法的时间复杂度为 O(n),空间复杂度为 O(n)。

例如,num = [2,3,4,2,6,2,5,1],size = 3,双向队列的变化情况为 dq: [2] -> [3] -> [4] -> [4,2] -> [6] (6 比 2 和 4 都大) -> [6,2] -> [6,5] (5 比 2 大) -> [5,1] (6 超出窗口,从队列首部出队)。

# -*- coding:utf-8 -*-
import collections
class Solution:
    def maxInWindows(self, num, size):
        # write code here
        if size > len(num) or size < 1:
            return []
        dq = collections.deque()  # [num[i], i]
        ans = []
        for i in range(len(num)):
            if dq and i - dq[0][1] >= size:
                dq.popleft()  # O(1)
            while dq and num[i] > dq[-1][0]: # 从后往前删除比 num[i] 大的数
                dq.pop()  # O(1)
            dq.append([num[i], i])
            if i >= size - 1:
                ans.append(dq[0][0])  # 队列首部始终最大
        return ans

剑指 offer 终于过了一遍,大多数题目还是很简单的,但是题目具有代表性,涉及链表、数组、深搜回溯、字符串、数组、数学、位运算、动态规划等。这里做一个总结:

剑指offer【03~09】
剑指offer【10~19】
剑指offer【20~29】
剑指offer【30~39】
剑指offer【40~49】
剑指offer【50~59】
剑指offer【60~68】

你可能感兴趣的:(剑指offer【50~59】)