《剑指 Offer》(第 2 版) 题解(Python 语言实现)第 51-60 题


文章目录

    • @[toc]
      • 第 50-1 题:字符串中第一个只出现一次的字符
      • 第 50-2 题:字符流中第一个不重复的字符
      • 第 51 题:数组中的逆序对
      • 第 52 题:两个链表的第一个公共结点
      • 第 53 题:数字在排序数组中出现的次数 (二分法典型问题)
      • 第 54 题:二叉搜索树的第 $k$ 大结点
      • 第 55-1 题:二叉树的深度
      • 第 55-2 题:平衡二叉树
      • 第 56-1 题:数组中只出现一次的两个数字
      • 0 到 n-1 中缺失的数字
      • 数组中数值和下标相等的元素
      • 第 56-2 题:数组中唯一只出现一次的数字
      • 第 57-1 题:和为 S 的两个数字
      • 第 57-2 题:和为 S 的连续正数序列
      • 第 58-1 题:翻转单词顺序列
      • 第 58-2 题:左旋转字符串
      • 第 59 题:滑动窗口的最大值(典型问题)
      • 第 60 题:$n$ 个骰子的点数(典型动态规划问题)

第 50-1 题:字符串中第一个只出现一次的字符

传送门:字符串中第一个只出现一次的字符。

在字符串中找出第一个只出现一次的字符。

如输入"abaccdeff",则输出b

如果字符串中不存在只出现一次的字符,返回#字符。

样例:

输入:"abaccdeff"

输出:'b'

同 LeetCode 第 387 题,传送门:字符串中的第一个唯一字符。

思路:特别容易想到的思路,就是统计词频,统计词频可以用哈希表,也可以用数组。

Python 代码:

class Solution:
    def firstNotRepeatingChar(self, s):
        """
        :type s: str
        :rtype: str
        """
        if len(s) < 1:
            return '#'

        counter = [0 for _ in range(256)]
        for alpha in s:
            counter[ord(alpha)] += 1
        for alpha in s:
            if counter[ord(alpha)] == 1:
                return alpha
        # 要注意:如果是 "aabbcc" 这种所有的字符都出现不止 1 次,
        # 就按照题意,返回 '#'
        return '#'

第 50-2 题:字符流中第一个不重复的字符

传送门:字符流中第一个只出现一次的字符。

请实现一个函数用来找出字符流中第一个只出现一次的字符。

例如,当从字符流中只读出前两个字符”go”时,第一个只出现一次的字符是’g’。

当从该字符流中读出前六个字符”google”时,第一个只出现一次的字符是’l’。

如果当前字符流没有存在出现一次的字符,返回#字符。

样例:

输入:"google"

输出:"ggg#ll"

解释:每当字符流读入一个字符,就进行一次判断并输出当前的第一个只出现一次的字符。

Python 代码:使用辅助数组,做字母频数统计

class Solution:

    def __init__(self):
        self.chars = [0 for _ in range(256)]
        self.strs = []

    def firstAppearingOnce(self):
        """
        :rtype: str
        """
        for char in self.strs:
            if self.chars[ord(char)] == 1:
                return char
        return '#'

    def insert(self, char):
        """
        :type char: str
        :rtype: void
        """
        self.chars[ord(char)] += 1
        self.strs.append(char)

Java 代码:

Python 代码:

Python 代码:

class Solution:
    h = {}
    r = []
    def firstAppearingOnce(self):
        while len(Solution.r):
            if Solution.h[Solution.r[0]] == 1:
                return Solution.r[0]
            Solution.r = Solution.r[1:]
        return '#'

    def insert(self, char):
        Solution.h[char] = Solution.h.get(char, 0) + 1
        if Solution.h[char] == 1:
            Solution.r.append(char)

作者:applezjm
链接:https://www.acwing.com/activity/content/code/content/19320/
来源:AcWing
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

第 51 题:数组中的逆序对

传送门:数组中的逆序对。

在数组中的两个数字如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对。

输入一个数组,求出这个数组中的逆序对的总数。

样例:

输入:[1,2,3,4,5,6,0]

输出:6

专门整理成文章。1、用归并排序;2、用 BST。如何记录左子树中结点的个数。

第 52 题:两个链表的第一个公共结点

传送门:两个链表的第一个公共结点。

输入两个链表,找出它们的第一个公共结点。

样例:

给出两个链表如下所示:

A:    a1 → a2
                  ↘
                    c1 → c2 → c3
                  ↗            
B:b1 → b2 → b3

输出第一个公共节点 c1。

思路1:两个链表如果有相同起点的话就好办了,所以首先要计算出两个链表的长度,进而计算它们的差值。

Python 代码:

class ListNode(object):
    def __init__(self, x):
        self.val = x
        self.next = None


class Solution(object):

    def __get_list_node_size(self, root):
        node = root
        size = 0
        while node:
            size += 1
            node = node.next
        return size

    def findFirstCommonNode(self, headA, headB):
        """
        :type headA, headB: ListNode
        :rtype: ListNode
        """
        if headA is None or headB is None:
            return None

        s1 = self.__get_list_node_size(headA)
        s2 = self.__get_list_node_size(headB)

        # 我们默认 l1 >= l2
        h1 = headA
        h2 = headB

        if s2 > s1:
            # 如果 B 长度更长,把二者交换
            h1 = headB
            h2 = headA
        # 现在 h1 上走 (s1 - s2) 这么多长度
        for _ in range(abs(s1 - s2)):
            h1 = h1.next
        # 然后齐头并进
        while h1 and h2 and h1.val != h2.val:
            h1 = h1.next
            h2 = h2.next

        # 走到这里,如果是因为 h1 和 h2 都空了,返回 Node
        if h1 is None and h2 is None:
            return None
        else:
            return h1

思路2:用两个栈。

Python 代码:写法上要注意,不要想当然

class ListNode(object):
    def __init__(self, x):
        self.val = x
        self.next = None


class Solution(object):

    def findFirstCommonNode(self, headA, headB):
        """
        :type headA, headB: ListNode
        :rtype: ListNode
        """
        if headA is None or headB is None:
            return None
        stack1 = []
        stack2 = []
        node1 = headA
        while node1:
            stack1.append(node1)
            node1 = node1.next
        node2 = headB
        while node2:
            stack2.append(node2)
            node2 = node2.next
        # 注意:这里有陷阱,一定要先设置一个 result 结点
        # 如果两个链表没有公共元素,res 不会被赋值
        res = None
        while stack1 and stack2:
            node1 = stack1.pop()
            node2 = stack2.pop()
            if node1.val == node2.val:
                # 这里暂存一下,最后一个相等的结点才是我们求的
                res = node1
                continue
            if stack1 is None or stack2 is None:
                return None
        return res

思路3:拼成一样长,这个写法记住就可以了。

Python 代码:

class Solution(object):

    def findFirstCommonNode(self, headA, headB):
        """
        :type headA, headB: ListNode
        :rtype: ListNode
        """
        if headA is None or headB is None:
            return None
        p1 = headA
        p2 = headB

        while p1 != p2:
            if p1 is None:
                p1 = headB
            else:
                p1 = p1.next
            if p2 is None:
                p2 = headA
            else:
                p2 = p2.next
        return p1

第 53 题:数字在排序数组中出现的次数 (二分法典型问题)

传送门:数字在排序数组中出现的次数。

统计一个数字在排序数组中出现的次数。

例如输入排序数组 [1, 2, 3, 3, 3, 3, 4, 5] 和数字 3 ,由于 3 在这个数组中出现了 4 次,因此输出4。

样例:

输入:[1, 2, 3, 3, 3, 3, 4, 5] , 3

输出:4

参考资料:《剑指 Offer》(第 2 版)第 53 题:数字在排序数组中出现的次数。

思路1:写一个二分法,使用二分法找到大于等于 k k k 的第 1 1 1 个数的下标,再使用二分法找到大于等于 k + 1 k+1 k+1 的第 1 1 1 个数的下标,二者之差即为所求。特别注意,这里是如何使用二分法的。

Python 代码:

class Solution(object):

    # 返回大于等于 target 的第 1 个数
    def get_left(self, nums, target):
        # [2,3,4,5,5,5,5,5,5,5]
        # [1,1,1,1,1,1,1,1,1,2,3,4,5,5,5,5,5,5,5]
        if nums[0] == target:
            return 0
        l = 1
        r = len(nums)
        while l < r:
            mid = l + (r - l) // 2
            if nums[mid] < target:
                l = mid + 1
            else:
                assert nums[mid] >= target
                # 不能排除 mid
                r = mid
        return l

    def getNumberOfK(self, nums, k):
        """
        :type nums: list[int]
        :type k: int
        :rtype: int
        """
        size = len(nums)
        if size == 0:
            return 0

        return self.get_left(nums, k + 1) - self.get_left(nums, k)

严格按照二分法模板的话,代码要这样写:

Python 代码:

class Solution(object):

    def getNumberOfK(self, nums, k):
        """
        :type nums: list[int]
        :type k: int
        :rtype: int
        """
        size = len(nums)
        if size == 0:
            return 0

        # 设置辅助函数,给一个 nums,一个 k,返回大于等于 k 的第一个数的索引
        return self.__helper(nums, k + 1) - self.__helper(nums, k)

    def __helper(self, nums, k):
        """
        返回大于等于 k 的第一个数的索引
        :param nums:
        :param k:
        :return:
        """
        size = len(nums)
        if size == 0:
            return 0

        l = 0
        # 注意:这里一定要写 size
        r = size - 1
        while l < r:
            mid = l + (r - l) // 2
            if nums[mid] >= k:
                r = mid
            else:
                assert nums[mid] < k
                # [1,2,3,4,5]
                l = mid + 1
        # 因为 k 有可能不存在,所以不一定符合要求,所以一定要单独判断一下
        if nums[l] != k:
            if nums[size - 1] < k:
                return size
            elif nums[0] > k:
                return 0
        return l

C++ 代码:

class Solution {
public:
    int getNumberOfK(vector& nums , int k) {
        if (nums.empty()) return 0;
        return helper(nums, k + 1) - helper(nums, k);
    }

    int helper(vector& nums, int k){
        int l = 0, r = nums.size();
        while (l < r){
            int m = l + (r - l) / 2;
            if (nums[m] < k) l = m + 1;
            else r = m;
        }
        return l;
    }
};

思路2:写两个二分法,一个数出现的次数,一个数最右边的索引 - 一个数最左边的索引 + 1。

# # 56、数字在排序数组中出现的次数
# 统计一个数字在排序数组中出现的次数。
#
# 例如输入排序数组[1, 2, 3, 3, 3, 3, 4, 5]和数字3,由于3在这个数组中出现了4次,因此输出4。


class Solution(object):

    def getNumberOfK(self, nums, k):
        """
        :type nums: list[int]
        :type k: int
        :rtype: int
        """
        size = len(nums)
        if size == 0:
            return 0

        # 设置辅助函数,给一个 nums,一个 k,返回大于等于 k 的第一个数的索引

        k_right = self.__get_right_k(nums, k)
        k_left = self.__get_left_k(nums, k)

        if k_right == -1 or k_left == -1:
            return 0

        return k_right - k_left + 1

    def __get_right_k(self, nums, k):
        # 找到最右边的 index ,使得 nums[index] = k
        size = len(nums)
        if size == 0:
            return -1
        l = 0
        r = size - 1
        while l < r:
            mid = l + (r - l + 1) // 2
            if nums[mid] <= k:
                # [1,2,5,5,5,7]
                l = mid
            elif nums[mid] > k:
                r = mid - 1
        if nums[l] != k:
            return -1
        return l

    def __get_left_k(self, nums, k):
        # 找到最左边的 index ,使得 nums[index] = k
        size = len(nums)
        if size == 0:
            return -1
        l = 0
        r = size - 1
        while l < r:
            mid = l + (r - l) // 2
            if nums[mid] >= k:
                r = mid
            else:
                assert nums[mid] < k
                l = mid + 1
        if nums[l] != k:
            return -1
        return l


if __name__ == '__main__':
    nums = [2, 3, 4, 5, 5, 5, 5, 5, 5, 5, 6, 7, 8, 9, 10]
    k = 5
    solution = Solution()
    # result = solution.get_left(nums, 5, )
    # print(result)

    result = solution.getNumberOfK(nums, k)
    print(result)

第 54 题:二叉搜索树的第 k k k 大结点

传送门:二叉搜索树的第 k 大结点。

给定一棵二叉搜索树,请找出其中的第 k k k 小的结点。

你可以假设树和 k k k 都存在,并且 1≤ k ≤ 树的总结点数。

样例:

输入:root = [2, 1, 3, null, null, null, null]k = 3

    2
  / \
1   3

输出:3

思路:使用栈模拟 BST 的中序遍历。

Python 代码:

class Solution(object):
    def kthNode(self, root, k):
        """
        :type root: TreeNode
        :type k: int
        :rtype: TreeNode
        """

        if root is None:
            return None

        # 1 表示递归处理,0 表示当前我就要处理这个结点
        stack = [(1, root)]

        while stack:
            type, node = stack.pop()
            if type == 0:
                k -= 1
                if k == 0:
                    return node
            else:
                if node.right:
                    stack.append((1, node.right))
                stack.append((0, node))
                if node.left:
                    stack.append((1, node.left))

第 55-1 题:二叉树的深度

传送门:二叉树的深度。

输入一棵二叉树的根结点,求该树的深度。

从根结点到叶结点依次经过的结点(含根、叶结点)形成树的一条路径,最长路径的长度为树的深度。

样例:

输入:二叉树 [8, 12, 2, null, null, 6, 4, null, null, null, null] 如下图所示:

      8
     / \
   12  2
       / \
     6   4

输出:3

思路:使用广度优先遍历。

Python 代码:

class Solution:
    def treeDepth(self, root):
        """
        :type root: TreeNode
        :rtype: int
        """

        if root is None:
            return 0

        queue = [(1, root)]
        res = 0
        while queue:
            top = queue.pop(0)
            cur_depth, node = top[0], top[1]
            res = max(res, cur_depth)
            if node.left:
                queue.append((cur_depth + 1, node.left))
            if node.right:
                queue.append((cur_depth + 1, node.right))
        return res

第 55-2 题:平衡二叉树

传送门:平衡二叉树。

输入一棵二叉树的根结点,判断该树是不是平衡二叉树。

如果某二叉树中任意结点的左右子树的深度相差不超过1,那么它就是一棵平衡二叉树。

注意:

  • 规定空树也是一棵平衡二叉树。

样例:

输入:二叉树 [5,7,11,null,null,12,9,null,null,null,null] 如下所示,

      5
    / \
  7  11
    / \
  12  9

输出:true

思路:深度优先遍历。

Python 代码:

# Definition for a binary tree node.
# class TreeNode(object):
#     def __init__(self, x):
#         self.val = x
#         self.left = None
#         self.right = None

class Solution(object):
    flag = 1

    def isBalanced(self, root):
        """
        :type root: TreeNode
        :rtype: bool
        """

        if root is None:
            return True
        self.__dfs(root)
        return self.flag

    def __dfs(self, node):
        """
        返回以 root 为根的二叉树的高度,如果左右子树其中之一不是 AVL ,则返回 -1
        :param node:
        :return:
        """
        if node is None:
            return 0
        left = self.__dfs(node.left)
        right = self.__dfs(node.right)

        if abs(left - right) > 1:
            self.flag = 0
            # 这里不能写 return
        return max(left, right) + 1

第 56-1 题:数组中只出现一次的两个数字

传送门:数组中只出现一次的两个数字。

一个整型数组里除了两个数字之外,其他的数字都出现了两次。

请写程序找出这两个只出现一次的数字。

你可以假设这两个数字一定存在。

样例:

输入: [ 1 , 2 , 3 , 3 , 4 , 4 ] [1,2,3,3,4,4] [1,2,3,3,4,4]

输出: [ 1 , 2 ] [1,2] [1,2]

思路:按位分组

Python 代码:

class Solution(object):
    def findNumsAppearOnce(self, nums):
        """
        :type nums: List[int]
        :rtype: List[int]
        """
        l = len(nums)
        if l < 2:
            raise Exception('程序出错')
        if l == 2:
            return nums

        # 全部相与一遍
        xor = 0
        for num in nums:
            xor ^= num
            
        # 最末尾的 1 从右向左边数在第几位
        counter = 0
        while xor & 1 == 0:
            xor >>= 1
            counter += 1

        res = [0, 0]
        for num in nums:
            if (num >> counter) & 1 == 1:
                res[1] ^= num
            else:
                res[0] ^= num
        return res

Java 代码:

import java.util.Arrays;

// 第 56 题:数组中数字出现的次数 P275
// 参考资料:
// 1、https://blog.csdn.net/derrantcm/article/details/46771717
public class Solution {

    // 考察位运算:或、与、异或、非,以及无符号左移 >>>
    public int[] findNumbersAppearanceOnce(int[] nums) {
        int len = nums.length;
        int[] res = new int[2];
        assert len >= 2;
        if (len == 2) {
            return nums;
        }
        // 那两个只出现一次的数的异或运算的结果
        int xor = xor(nums);
        
        // 关键在这里
        // 找到这个 xor 的二进制表示第 1 个是 1 的数位是第几位
        int binaryFirstNotZero = binaryFirstNotZero(xor);
        
        // 接下来分别对两组进行异或
        for (int i = 0; i < len; i++) {
            // 如果这个数右移这么多位是 1 的分在一组,是 0 的分在另外一组,遍历的时候,就进行异或运算
            if ((nums[i] >>> binaryFirstNotZero & 1) == 1) {
                res[0] ^= nums[i];
            } else {
                res[1] ^= nums[i];
            }
        }
        return res;
    }

    // 得到一个数组经过异或运算的结果 xor
    // 异或 的英文翻译就是 xor
    private int xor(int[] nums) {
        int xor = 0;
        for (int i = 0; i < nums.length; i++) {
            xor ^= nums[i];
        }
        return xor;
    }

    // 得到一个整数的二进制表示从右到左第 1 个非零的位数是第几位
    private int binaryFirstNotZero(int num) {
        int index = 0;
        // 这里的 1 把它看成二进制的 1,即 00000001
        while ((num & 1) == 0 && index < 32) {
            num >>>= 1;
            index++;
        }
        // 走到这里满足 (num & 1) == 1
        return index;
    }

    public static void main(String[] args) {
        int[] nums = {2, 4, 3, 6, 3, 2, 5, 5};
        Solution solution = new Solution();
        int[] res = solution.findNumbersAppearanceOnce(nums);
        System.out.println(Arrays.toString(res));

        int[] nums2 = {2, 4, 3, 6, 3, 2, 5, 5};
        int[] res2 = solution.findNumbersAppearanceOnce(nums2);
        System.out.println(Arrays.toString(res2));

        int[] nums3 = {4, 6};
        int[] res3 = solution.findNumbersAppearanceOnce(nums3);
        System.out.println(Arrays.toString(res3));

        int[] nums4 = {4, 6, 1, 1, 1, 1};
        int[] res4 = solution.findNumbersAppearanceOnce(nums4);
        System.out.println(Arrays.toString(res4));
    }
}

0 到 n-1 中缺失的数字

传送门:0 到 n-1 中缺失的数字。

一个长度为 n-1 的递增排序数组中的所有数字都是唯一的,并且每个数字都在范围 0 到 n-1 之内。

在范围 0 到 n-1 的 n 个数字中有且只有一个数字不在该数组中,请找出这个数字。

样例

输入:[0,1,2,4]

输出:3

思路:典型的使用“二分法”解决的问题。

Python 代码:

class Solution(object):
    def getMissingNumber(self, nums):
        """
        :type nums: List[int]
        :rtype: int
        """

        size = len(nums)
        l = 0
        r = size

        while l < r:

            mid = l + (r - l) // 2
            if nums[mid] > mid:
                # [0,1,2,3,4,6]
                # mid 有可能是要求的数
                r = mid
            else:
                assert nums[mid] <= mid
                l = mid + 1
        return l


if __name__ == '__main__':
    solution = Solution()
    nums = [0, 1, 2, 4]
    result = solution.getMissingNumber(nums)
    print(result)

数组中数值和下标相等的元素

传送门:数组中数值和下标相等的元素。

假设一个单调递增的数组里的每个元素都是整数并且是唯一的。

请编程实现一个函数找出数组中任意一个数值等于其下标的元素。

例如,在数组 [ − 3 , − 1 , 1 , 3 , 5 ] [-3, -1, 1, 3, 5] [3,1,1,3,5] 中,数字 3 3 3 和它的下标相等。

样例:

输入: [ − 3 , − 1 , 1 , 3 , 5 ] [-3, -1, 1, 3, 5] [3,1,1,3,5]

输出: 3 3 3

注意:如果不存在,则返回 − 1 -1 1

思路:典型的使用“二分法”解决的问题。

Python 代码:

class Solution(object):
    def getNumberSameAsIndex(self, nums):
        """
        :type nums: List[int]
        :rtype: int
        """

        # 使用二分法
        size = len(nums)
        l = 0
        r = size - 1

        while l < r:
            mid = l + (r - l) // 2
            if nums[mid] < mid:
                l = mid + 1
            else:
                assert nums[mid] >= mid
                r = mid
        return l if nums[l] == l else -1

第 56-2 题:数组中唯一只出现一次的数字

传送门:数组中唯一只出现一次的数字。

在一个数组中除了一个数字只出现一次之外,其他数字都出现了三次。

请找出那个只出现一次的数字。

你可以假设满足条件的数字一定存在。

思考题:

  • 如果要求只使用 O ( n ) O(n) O(n) 的时间和额外 O ( 1 ) O(1) O(1) 的空间,该怎么做呢?

样例:

输入:[1,1,1,2,2,2,3,4,4,4]

输出:3

思路:限制在 O ( 1 ) O(1) O(1) 空间复杂度,那就只有通过二进制,一位一位去看了。

Python 代码:

class Solution(object):
    def findNumberAppearingOnce(self, nums):
        """
        :type nums: List[int]
        :rtype: int
        """

        res = 0

        for i in range(32):
            count = 0
            for num in nums:
                # 不要忘记 & 1
                if (num >> i) & 1:
                    count += 1
            if count % 3:
                res += 1 << i
        return res


if __name__ == '__main__':
    nums = [1, 0, 0, 0, 2, 1, 1]
    solution = Solution()
    result = solution.findNumberAppearingOnce(nums)
    print(result)

第 57-1 题:和为 S 的两个数字

传送门:AcWing:和为 S 的两个数字。

输入一个数组和一个数字 s ,在数组中查找两个数,使得它们的和正好是s。

如果有多对数字的和等于 s ,输出任意一对即可。

你可以认为每组输入中都至少含有一组满足条件的输出。

样例:

输入:[1,2,3,4] , sum=7

输出:[3,4]

说明:同 LeetCode 第 1 题,必须要会的一题。

Python 代码:

class Solution(object):
    def findNumbersWithSum(self, nums, target):
        """
        :type nums: List[int]
        :type target: int
        :rtype: List[int]
        """
        
        s = set()
        for num in nums:
            if target - num not in s:
                s.add(num)
            else:
                return [num, target - num]

第 57-2 题:和为 S 的连续正数序列

传送门:AcWing:和为 S 的连续正数序列。

输入一个正数 s ,打印出所有和为 s 的连续正数序列(至少含有两个数)。

例如输入 15 15 15,由于 1 + 2 + 3 + 4 + 5 = 4 + 5 + 6 = 7 + 8 = 15 1+2+3+4+5=4+5+6=7+8=15 1+2+3+4+5=4+5+6=7+8=15,所以结果打印出 3 3 3 个连续序列 1 ~ 5 1~5 15 4 ~ 6 4~6 46 7 ~ 8 7~8 78

样例:

输入: 15 15 15

输出: [ [ 1 , 2 , 3 , 4 , 5 ] , [ 4 , 5 , 6 ] , [ 7 , 8 ] ] [[1,2,3,4,5],[4,5,6],[7,8]] [[1,2,3,4,5],[4,5,6],[7,8]]

思路:双指针,因为是有序数组,所以可以使用二分法。

设定两个指针,一个指向第一个数,一个指向最后一个数,在此之前需要设定第一个数和最后一个数的值,由于是正数序列,所以可以把第一个数设为 1 1 1,最后一个数为 2 2 2,因为是要求是连续正数序列,最后不可能和第一个数重合。下一步就是不断改变第一个数和最后一个数的值,如果从第一个数到最后一个数的和刚好是要求的和,那么把所有的数都添加到一个序列中;如果大于要求的和,则说明从第一个数到最后一个数之间的范围太大,因此减小范围,需要把第一个数的值加 1 1 1,同时把当前和减去原来的第一个数的值;如果小于要求的和,说明范围太小,因此把最后一个数加 1 1 1,同时把当前的和加上改变之后的最后一个数的值。这样,不断修改第一个数和最后一个数的值,就能确定所有连续正数序列的和等于 S S S 的序列了。

等差数列求和公式,首项加末项的和乘以个数除以 2 2 2,即 s u m = ( a + b ) × n 2 {\rm sum} = \cfrac {(a + b)\times n }{2} sum=2(a+b)×n

注意:右边界问题,使用一个特例,例如 3 3 3 就可以考虑清楚了。

Python 代码:

class Solution(object):
    def findContinuousSequence(self, sum):
        """
        :type sum: int
        :rtype: List[List[int]]
        """

        res = []

        left = 1
        right = 2

        # sum = 3 的时候,右边界最多到 2
        half = sum // 2 + 1
        while left < right <= half:
            cur_sum = (left + right) * (right - left + 1) // 2
            if cur_sum == sum:
                res.append([i for i in range(left, right + 1)])
                right += 1
            elif cur_sum < sum:
                right += 1
            else:
                assert cur_sum > sum
                left += 1
        return res


if __name__ == '__main__':
    sum = 15
    solution = Solution()
    result = solution.findContinuousSequence(sum)
    print(result)

第 58-1 题:翻转单词顺序列

传送门:AcWing:翻转单词顺序。

输入一个英文句子,翻转句子中单词的顺序,但单词内字符的顺序不变。

为简单起见,标点符号和普通字母一样处理。

例如输入字符串"I am a student.",则输出"student. a am I"

样例:

输入:"I am a student."

输出:"student. a am I"

思路:实现一个辅助的方法,将一个字符数组的指定区间进行翻转。

Python 代码:

class Solution(object):
    def reverseWords(self, s):
        """
        :type s: str
        :rtype: str
        """

        size = len(s)
        arr = list(s)

        self.__reverse(arr, 0, size - 1)

        begin = 0
        index = 0
        while index < size:
            if arr[index] == ' ':
                self.__reverse(arr, begin, index - 1)
                begin = index + 1
            index += 1
        # 最后还要反转一下
        self.__reverse(arr, begin, size - 1)
        return ''.join(arr)

    def __reverse(self, arr, left, right):
        if left >= right:
            return
        while left < right:
            arr[left], arr[right] = arr[right], arr[left]
            left += 1
            right -= 1

总结:别把问题想复杂了,有点耐心,这些问题其实并没有那么难;

第 58-2 题:左旋转字符串

传送门:左旋转字符串。

字符串的左旋转操作是把字符串前面的若干个字符转移到字符串的尾部。

请定义一个函数实现字符串左旋转操作的功能。

比如输入字符串"abcdefg"和数字2,该函数将返回左旋转2位得到的结果"cdefgab"

注意:

  • 数据保证n小于等于输入字符串的长度。

样例

输入:"abcdefg" , n=2

输出:"cdefgab"

思路:“abcdefg” 旋转以后,“gfedcba”,在倒数第 2 位前后,再翻转一下,得“cdefgab”。左旋转字符串的时候要分析清楚,注意:移动的位数要取余数。

Python 代码:

class Solution(object):
    def leftRotateString(self, s, n):
        """
        :type s: str
        :type n: int
        :rtype: str
        """

        size = len(s)
        # 特判
        if size == 0 or n % size == 0:
            return s
        n = n % size
        arr = list(s)
        self.__reverse(arr, 0, size - 1)

        self.__reverse(arr, 0, size - 1 - n)
        self.__reverse(arr, size - n, size - 1)

        return ''.join(arr)

    def __reverse(self, arr, left, right):
        if left >= right:
            return
        while left < right:
            arr[left], arr[right] = arr[right], arr[left]
            left += 1
            right -= 1

第 59 题:滑动窗口的最大值(典型问题)

传送门:滑动窗口的最大值。

给定一个数组和滑动窗口的大小,请找出所有滑动窗口里的最大值。

例如,如果输入数组 [2, 3, 4, 2, 6, 2, 5, 1] 及滑动窗口的大小 3,那么一共存在 6 个滑动窗口,它们的最大值分别为 [4, 4, 6, 6, 6, 5]

注意:

  • 数据保证 k k k 大于 0 0 0 ,且 k k k 小于等于数组长度。

样例:

输入:[2, 3, 4, 2, 6, 2, 5, 1] , k=3

输出: [4, 4, 6, 6, 6, 5]

同 LeetCode 第 239 题,传送门:LeetCode 第 239 题:滑动窗口的最大值。

Python 代码:window[0] == i - k 这个条件特别容易忽略,表示确实该被移出滑动窗口

class Solution:
    def maxSlidingWindow(self, nums, k):
        """
        :type nums: List[int]
        :type k: int
        :rtype: List[int]
        """

        # 关键:如果后进来一个数,前面的元素比它小
        # 那么前面的元素就永远不可能是"滑动窗口中的最大值"

        l = len(nums)
        if l == 0 or k <= 0:
            return []

        res = []
        window = []
        for i in range(l):

            # 考虑什么时候,要把最大移除
            # 左边界划出的时候,应该是 window.pop(0)
            # [0,1,2,3,4]
            #     [    i]
            # window[0] == i - k 这个条件特别容易忽略
            if i >= k and window[0] == i - k:
                window.pop(0)
            # 考虑把不可能是最大的元素全部 kill 掉
            while window and nums[i] >= nums[window[-1]]:
                window.pop()
            # 不管怎么着都加当前的索引
            window.append(i)

            # 什么时候有滑动窗口呢?
            if i >= k - 1:
                res.append(nums[window[0]])
        return res


if __name__ == '__main__':
    nums = [1, 3, -1, -3, 5, 3, 6, 7]
    k = 3

    solution = Solution()
    result = solution.maxSlidingWindow(nums, k)
    print(result)

第 60 题: n n n 个骰子的点数(典型动态规划问题)

传送门:AcWing:骰子的点数。

将一个骰子投掷 n n n 次,获得的总点数为 s s s s s s 的可能范围为 n n n ~ 6 n 6n 6n

掷出某一点数,可能有多种掷法,例如投掷 2 2 2 次,掷出 3 3 3 点,共有 [ 1 , 2 ] [1,2] [1,2] [ 2 , 1 ] [2,1] [2,1] 两种掷法。

请求出投掷 n n n 次,掷出 n n n ~ 6 n 6n 6n 点分别有多少种掷法。

样例1:

输入:n=1

输出:[1, 1, 1, 1, 1, 1]

解释:投掷 1 次,可能出现的点数为 1-6 ,共计 6 种。每种点数都只有 1 种掷法。所以输出 [1, 1, 1, 1, 1, 1]

样例2:

输入:n=2

输出:[1, 2, 3, 4, 5, 6, 5, 4, 3, 2, 1]

解释:投掷 2 次,可能出现的点数为 2-12,共计 11 种。每种点数可能掷法数目分别为 1,2,3,4,5,6,5,4,3,2,1。所以输出 [1, 2, 3, 4, 5, 6, 5, 4, 3, 2, 1]

思路:典型动态规划问题。定义状态 dp[i][j] 表示用 i 个骰子扔出和为 j 的可能数,因为第 i 个骰子可能扔出 1-6 的点数。根据此写出状态转移方程:
d p [ i ] [ j ] = d p [ i − 1 ] [ j − 1 ] + d p [ i − 1 ] [ j − 2 ] + d p [ i − 1 ] [ j − 3 ] + d p [ i − 1 ] [ j − 4 ] + d p [ i − 1 ] [ j − 5 ] + d p [ i − 1 ] [ j − 6 ] dp[i][j]=dp[i-1][j-1]+dp[i-1][j-2]+dp[i-1][j-3]+dp[i-1][j-4]+dp[i-1][j-5]+dp[i-1][j-6] dp[i][j]=dp[i1][j1]+dp[i1][j2]+dp[i1][j3]+dp[i1][j4]+dp[i1][j5]+dp[i1][j6]
由于我们只需要用到最后一次的结果,因此为了节省空间可以使用滚动数组,将二维 dp 数组变为一维。

时间复杂度分析: O ( n 2 ) O(n^2) O(n2)

Python 代码:

class Solution(object):
    def numberOfDice(self, n):
        """
        :type n: int
        :rtype: List[int]
        """
        dp = [0 for _ in range(6 * n + 1)]

        # 动归数组初始值,表示 1 个骰子扔出 1-6 的可能数都为 1
        for i in range(1, 7):
            dp[i] = 1
        # 表示仍第 2 个骰子到第 n 个骰子
        for i in range(2, n + 1):
            # 从后向前写
            for j in range(6 * i, -1, -1):
                dp[j] = 0
                # 最后一个骰子可以扔 1 - 6 点
                for k in range(6, 0, -1):
                    if j - k < 0:
                        continue
                    dp[j] += dp[j - k]
        # 扔 n 个骰子的和为 [n, 6 * n]
        return dp[n:]

作者:cornerCao
链接:https://www.acwing.com/solution/acwing/content/852/
来源:AcWing
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

(本节完)

你可能感兴趣的:(算法,数据结构,算法,数据结构)