LeetCode刷题笔记(算法思想 一)

LeetCode刷题笔记(算法思想 一)

  • 一、双指针
      • 167. 两数之和 II - 输入有序数组
      • 633. 平方数之和
      • 345. 反转字符串中的元音字母
      • 680. 验证回文字符串 Ⅱ
      • 88. 合并两个有序数组
      • 141. 环形链表
      • 142. 环形链表 II
      • 524. 通过删除字母匹配到字典里最长单词
  • 二、排序
      • 215. 数组中的第K个最大元素
      • 347. 前 K 个高频元素
      • 451. 根据字符出现频率排序
      • 75. 颜色分类

一、双指针

167. 两数之和 II - 输入有序数组

给定一个已按照升序排列的有序数组,找到两个数使得它们相加之和等于目标数。
函数应该返回这两个下标值 index1 和 index2,其中 index1 必须小于 index2。

说明:

  • 返回的下标值(index1 和 index2)不是从零开始的。
  • 你可以假设每个输入只对应唯一的答案,而且你不可以重复使用相同的元素。

示例:

输入: numbers = [2, 7, 11, 15], target = 9
输出: [1,2]
解释: 2 与 7 之和等于目标数 9 。因此 index1 = 1, index2 = 2 。

思路: 定义两个指针,i 从数组开头向尾遍历,j 从数组尾向开头遍历,每次将指针指向的两数之和与target比较,比target大则 j 向前移,比target小则 i 向后移,和target相等则返回。注意题目要求下标不是从0开始。

另外本题还可以用二分查找法实现:循环遍历数组,由于target已知,target与数组中a的差即为要找的另一个数b。利用二分法可以快速查找b,最后返回排序好的a,b的下标。

代码实现:

# 双指针解法:
class Solution:
    def twoSum(self, nums, target):
        i, j = 0, len(nums)-1
        while i < j:
            if nums[i] + nums[j] > target:
                j -= 1
            elif nums[i] + nums[j] < target:
                i += 1
            elif nums[i] + nums[j] == target:
                return sorted([i+1,j+1])   # 下标不从0开始
        return None
# 二分查找解法:
class Solution:
    def binary_search(self, li, left, right, val):
        while left <= right:
            mid = (left + right) // 2
            if li[mid] > val:
                right = mid - 1
            elif li[mid] < val:
                left = mid + 1
            else:
                return mid
        return None

    def twoSum(self, nums, target):
        for i in range(len(nums)):
            a = nums[i]
            b = target - a
            if b >= a:
                j = self.binary_search(nums,i+1,len(nums)-1,b)
            else:
                j = self.binary_search(nums,0,i-1,b)
            if j:
                break
        return sorted([i+1,j+1])

633. 平方数之和

给定一个非负整数 c ,你要判断是否存在两个整数 a 和 b,使得 a^2 + b^2 = c。

示例1:

输入: 5
输出: True
解释: 1 * 1 + 2 * 2 = 5

示例2:

输入: 3
输出: False

思路:(1)双指针方法:由于a^2 + b^2 = c,a和b均小于c的平方根。设定两个指针,一个p从0开始往右扫描,一个r从int(c**0.5)开始往左扫描,循环结束条件是当p > r时则结束循环。循环里,当 p^2 + r^2 == c时,返回True;当p^2 + r^2 > c时,说明r选的大了,于是r -=1;当p^2 + r^2 < c时,说明p选的小了,于是 p += 1;循环结束还没找到,返回False。

(2)对上面方法改进,在循环里,b = math.sqrt(c - a*a),如果b是整数,b == int(b),直接返回True;循环结束还没找到,返回False。

代码实现:

import math
class Solution:
    def judgeSquareSum(self, c: int) -> bool:
        a = 0
        while a*a <= c:
            b = math.sqrt(c - a*a)
            if b == int(b):
                return True
            a += 1
        return False

345. 反转字符串中的元音字母

编写一个函数,以字符串作为输入,反转该字符串中的元音字母。

示例 1:

输入: "hello"
输出: "holle"

示例 2:

输入: "leetcode"
输出: "leotcede"

说明: 元音字母不包含字母"y"。

思路: 定义一个列表存储十个元音字母(大小写),将字符串转为列表,定义两个指针,i从0开始往右扫描,j从列表最后一个元素往左扫描。循环结束条件为 i >= j,如果指针指向的两个元素均在列表里(均为元音字母),交换并把 i += 1,j -= 1;如果一个在一个不在,不在的那个指针移动至下一位置;如果都不在,则两个指针都移动至下一位置。

代码实现:

class Solution:
    def reverseVowels(self, s: str) -> str:
        li = list(s)
        vowel = ['a','e','i','o','u','A','E','I','O','U']
        i = 0
        j = len(li)-1
        while i < j:
            if li[i] in vowel and li[j] in vowel:
                li[i], li[j] = li[j], li[i]
                i += 1
                j -= 1
            elif li[i] in vowel and li[j] not in vowel:
                j -= 1
            elif li[j] in vowel and li[i] not in vowel:
                i += 1
            else:
                i += 1
                j -= 1
        return "".join(li)

680. 验证回文字符串 Ⅱ

给定一个非空字符串 s,最多删除一个字符。判断是否能成为回文字符串。

示例 1:

输入: "aba"
输出: True

示例 2:

输入: "abca"
输出: True
解释: 你可以删除c字符。

注意: 字符串只包含从 a-z 的小写字母。字符串的最大长度是50000。

思路: 如果字符串等于它的倒序,直接返回True。定义两个指针,i从0开始往右扫描,j从列表最后一个元素往左扫描。循环结束条件是 i >= j,如果 s[i] == s[j] ,两指针分别移到下一位置;如果 s[i] != s[j],就看除去 i 位置元素字符串是否回文,或者除去 j 位置元素字符串是否回文,两者有一个成立则返回True,两者都不成立返回False。

代码实现:

class Solution:
    def validPalindrome(self, s: str) -> bool:
        if s == s[::-1]:
            return True
        i = 0
        j = len(s)-1
        while i < j:
            if s[i] == s[j]:
                i += 1
                j -= 1
            else:
                res1 = s[:i]+s[i+1:]
                res2 = s[:j]+s[j+1:]
                if res1 == res1[::-1] or res2== res2[::-1]:
                    return True
                else:
                    return False
        return True

88. 合并两个有序数组

给你两个有序整数数组 nums1 和 nums2,请你将 nums2 合并到 nums1 中,使 nums1 成为一个有序数组。

说明:

  • 初始化 nums1 和 nums2 的元素数量分别为 m 和 n 。
  • 你可以假设 nums1 有足够的空间(空间大小大于或等于 m + n)来保存 nums2 中的元素。

示例:

输入:
nums1 = [1,2,3,0,0,0], m = 3
nums2 = [2,5,6],       n = 3

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

思路: 双指针方法:首先判断如果m=0,直接将数组1清空,把数组2 extend过去;两数组指针分别从m和n开始,从后往前遍历,把两者之间较大的从后往前放到数组1;注意当m=0而n != 0的情况。

代码实现:

class Solution:
    def merge(self, nums1: List[int], m: int, nums2: List[int], n: int) -> None:
        """
        Do not return anything, modify nums1 in-place instead.
        """
        if m == 0:
            nums1.clear()
            nums1.extend(nums2)
        while m != 0:
            if n > 0 and nums1[m-1] <= nums2[n-1]:
                nums1[m+n-1] = nums2[n-1]
                n -= 1
            else:
                nums1[m+n-1] = nums1[m-1]
                m -= 1
        while n != 0:
            nums1[m+n-1] = nums2[n-1]
            n -= 1

141. 环形链表

给定一个链表,判断链表中是否有环。

为了表示给定链表中的环,我们使用整数 pos 来表示链表尾连接到链表中的位置(索引从 0 开始)。 如果 pos 是 -1,则在该链表中没有环。

示例 1:

输入:head = [3,2,0,-4], pos = 1
输出:true
解释:链表中有一个环,其尾部连接到第二个节点。

LeetCode刷题笔记(算法思想 一)_第1张图片

示例 2:

输入:head = [1,2], pos = 0
输出:true
解释:链表中有一个环,其尾部连接到第一个节点。

LeetCode刷题笔记(算法思想 一)_第2张图片

示例 3:

输入:head = [1], pos = -1
输出:false
解释:链表中没有环。

在这里插入图片描述

进阶:你能用 O(1)(即,常量)内存解决此问题吗?

思路: 判断环形链表有一个巧妙的方法就是用到快慢指针,slow指针一次挪动一个节点,fast指针一次挪动两个节点,如果链表没有环,肯定是快指针先走完,两个指针不会相遇;如果链表有环,两个指针最终肯定会在环里绕圈,而两指针速度不同所以一定会相遇。循环结束条件是slow == fast,所以通常一开始都自定义一个节点ListNode(0),这个节点的下一个是head,将slow指向自定义节点而fast指向head,目的是错开俩指针位置以进入循环。此处代码不将slow指向head而fast指向head.next的原因是如果链表只有一个节点,那就没有head下一个为None就没有.next。另外要注意考虑边界情况,fast是否存在,fast.fast是否存在等。

代码实现:

class Solution:
    def hasCycle(self, head: ListNode) -> bool:
        slow = ListNode(0)
        slow.next = head
        fast = head
        while slow != fast:
            if fast and fast.next:
                fast = fast.next.next
            else:
                return False
            slow = slow.next
        return True

142. 环形链表 II

给定一个链表,返回链表开始入环的第一个节点。 如果链表无环,则返回 null。

为了表示给定链表中的环,我们使用整数 pos 来表示链表尾连接到链表中的位置(索引从 0 开始)。 如果 pos 是 -1,则在该链表中没有环。

说明:不允许修改给定的链表。

示例 1:

输入:head = [3,2,0,-4], pos = 1
输出:tail connects to node index 1
解释:链表中有一个环,其尾部连接到第二个节点。

LeetCode刷题笔记(算法思想 一)_第3张图片

示例 2:

输入:head = [1,2], pos = 0
输出:tail connects to node index 0
解释:链表中有一个环,其尾部连接到第一个节点。

LeetCode刷题笔记(算法思想 一)_第4张图片

示例 3:

输入:head = [1], pos = -1
输出:no cycle
解释:链表中没有环。

在这里插入图片描述
进阶:你是否可以不用额外空间解决此题?

思路: 如果链表有环,快慢指针一定会在环内相遇(相遇证明有环),要找到环的入口,还要进行第二轮,将fast指针重置为head位置,slow保持相遇位置不变,这时fast和slow都是一次走一步,两指针这次相遇的位置即为环的入口。

LeetCode刷题笔记(算法思想 一)_第5张图片
证明: 第二轮当fast从head出发,slow从第一轮相遇点出发,两指针步长均为1,再次相遇时的位置即为环的入口。假设环的长度为L,从head到环入口的长度为x,环入口到相遇点M的长度为y,从相遇点M到环入口的长度为z。假设slow从相遇点M出发,已经走了n圈(nL),则fast从head出发,也同样走了nL,若两指针在环入口相遇,问题转化为证明:x - nL(图中黄色圈出的部分) = z,即 x = nL + z。

第一轮相遇时,Sslow = x + y; Sfast = 2(x + y); 此时fast比slow多走了k个L (k>0),故有:2(x + y) = x + y + kL; => x = kL - y; => x = (k-1)L + z 得证。

代码实现:

class Solution:
    def detectCycle(self, head: ListNode) -> ListNode:
        slow, fast = head, head
        while True:
            if fast and fast.next:
                slow = slow.next
                fast = fast.next.next
                if slow == fast:
                    break
            else:
                return None

        fast = head
        while slow != fast:
            slow = slow.next
            fast = fast.next
        return slow

524. 通过删除字母匹配到字典里最长单词

给定一个字符串和一个字符串字典,找到字典里面最长的字符串,该字符串可以通过删除给定字符串的某些字符来得到。如果答案不止一个,返回长度最长且字典顺序最小的字符串。如果答案不存在,则返回空字符串。

示例 1:

输入:
s = "abpcplea", d = ["ale","apple","monkey","plea"]

输出: 
"apple"

示例 2:

输入:
s = "abpcplea", d = ["a","b","c"]

输出: 
"a"

说明:

所有输入的字符串只包含小写字母。
字典的大小不会超过 1000。
所有输入的字符串长度不会超过 1000。

思路: 定义两个指针分别遍历两个单词,当遍历字典里单词的指针扫描完一整个单词,说明这个单词有可能是结果,把它再和结果字符串进行比较,选择较长的一个单词,若长度相等,返回字典顺序最小的。注意字典顺序是指按照字母表顺序,比如aa

代码实现:

class Solution:
    def findLongestWord(self, s: str, d):
        res = ""
        for str_d in d:
            j = 0
            for i in range(len(s)):
                if j < len(str_d):
                    if s[i] == str_d[j]:
                        j += 1
                    if j == len(str_d):  # 表示指针j扫描完了d中某个字符串,说明这个字符串可能是要找的
                        # 字典顺序是指按照字母表顺序,比如aa
                        if (len(str_d) > len(res)) or (len(str_d) == len(res) and str_d < res):
                            res = str_d
        return res

二、排序

215. 数组中的第K个最大元素

在未排序的数组中找到第 k 个最大的元素。请注意,你需要找的是数组排序后的第 k 个最大的元素,而不是第 k 个不同的元素。

示例 1:

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

示例 2:

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

说明: 你可以假设 k 总是有效的,且 1 ≤ k ≤ 数组的长度。

思路: 直接利用Python内置heapq.nlargest(k, nums)返回前k大的元素。求前k大是建立k个元素的小根堆;求前k小是建立k个元素的大根堆。

代码实现:

import heapq
class Solution:
    def findKthLargest(self, nums, k):
        li = heapq.nlargest(k, nums)
        return li[-1]

347. 前 K 个高频元素

给定一个非空的整数数组,返回其中出现频率前 k 高的元素。

示例 1:

输入: nums = [1,1,1,2,2,3], k = 2
输出: [1,2]

示例 2:

输入: nums = [1], k = 1
输出: [1]

提示:

你可以假设给定的 k 总是合理的,且 1 ≤ k ≤ 数组中不相同的元素的个数。
你的算法的时间复杂度必须优于 O(n log n) , n 是数组的大小。
题目数据保证答案唯一,换句话说,数组中前 k 个高频元素的集合是唯一的。
你可以按任意顺序返回答案。

思路: 用哈希表来存储每个元素出现的次数,Python中字典就是根据哈希表实现的,所以新建一个空字典,以每个数作为键,以每个数出现的次数作为对应的值。按字典的值倒序排列,输出前两个的键。注意字典是无序的,排序之后要传给一个变量,这个变量是排好序的,变量的形式是列表里放着一个一个元组。

代码实现:

class Solution:
    def topKFrequent(self, nums, k: int):
        hash = {}
        for i in range(len(nums)):
            if nums[i] not in hash.keys():
                hash[nums[i]] = 1
            elif nums[i] in hash.keys():
                hash[nums[i]] += 1
        # print(hash)
        # 字典是无序的,排序之后要传给一个变量,这个变量是排好序的
        d_order = sorted(hash.items(),key=lambda x:x[1],reverse=True)
        # print(d_order)
        res = []
        for i in range(k):
            res.append(d_order[i][0])
        return res

451. 根据字符出现频率排序

给定一个字符串,请将字符串里的字符按照出现的频率降序排列。

示例 1:

输入:
"tree"

输出:
"eert"

解释:
'e'出现两次,'r'和't'都只出现一次。
因此'e'必须出现在'r'和't'之前。此外,"eetr"也是一个有效的答案。

示例 2:

输入:
"cccaaa"

输出:
"cccaaa"

解释:
'c'和'a'都出现三次。此外,"aaaccc"也是有效的答案。
注意"cacaca"是不正确的,因为相同的字母必须放在一起。

示例 3:

输入:
"Aabb"

输出:
"bbAa"

解释:
此外,"bbaA"也是一个有效的答案,但"Aabb"是不正确的。
注意'A'和'a'被认为是两种不同的字符。

思路: 利用Python自带的Counter统计每个字符出现的次数,按次数倒序排列并拼接字符串输出。也可以直接利用Counter类的 most_common(),结果就是按出现次数倒序排列的。

代码实现:

import collections
class Solution:
    def frequencySort(self, s: str) -> str:
        hash = collections.Counter(s)
        print(hash)
        d_order = sorted(hash.items(),key= lambda x:x[1],reverse=True)
        res = []
        for i in range(len(d_order)):
            res.append(d_order[i][0]*d_order[i][1])
        return "".join(res)

直接利用Counter类的 most_common(),一行搞定:

import collections
class Solution:
    def frequencySort(self, s: str) -> str:
        # Counter(s) 返回一个类似字典的结构,键是s中的项,值是这个项出现的次数
        # Counter类有一个函数most_common(int n),列出前n个元素,如果不加参数,就全部返回,相当于按照出现次数从大到小输出
        return "".join(k*v for k,v in collections.Counter(s).most_common())

75. 颜色分类

给定一个包含红色、白色和蓝色,一共 n 个元素的数组,原地对它们进行排序,使得相同颜色的元素相邻,并按照红色、白色、蓝色顺序排列。

此题中,我们使用整数 0、 1 和 2 分别表示红色、白色和蓝色。

注意: 不能使用代码库中的排序函数来解决这道题。

示例:

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

进阶:

一个直观的解决方案是使用计数排序的两趟扫描算法。
首先,迭代计算出0、1 和 2 元素的个数,然后按照0、1、2的排序,重写当前数组。
你能想出一个仅使用常数空间的一趟扫描算法吗?

思路: 此题是经典的荷兰国旗问题,使用三路快排方法效率较高。用两个指针zero,two将数组分为三个区间,用另一个指针p遍历整个数组。定义三个区间,排序完成时:[0,zero)都是0;[zero,two)都是1;[tow:]都是2。用指针p遍历整个数组,将每个元素放到正确的区间,通过移动zero和two,当p超过two时,数组刚好被遍历完一遍,并且是排好序的。

代码实现:

# 三路快排方法:
class Solution:
    def sortColors(self, nums):
        # 定义三个区间,排序完成时:[0,zero)都是0;[zero,two)都是1;[tow:]都是2
        # 用指针p遍历整个数组,将每个元素放到正确的区间,通过移动zero和two
        # 当p超过two时,数组刚好被遍历完一遍,并且是排好序的
        zero = 0
        two = len(nums)-1
        p = 0
        while p <= two:
            if nums[p] == 0:
                self.switch(nums, p, zero)
                p += 1
                zero += 1
            elif nums[p] == 1:
                p += 1
            else:  # nums[p] == 2
                self.switch(nums, p, two)
                two -=1
        print(nums)

    def switch(self, nums, i, j):
        nums[i], nums[j] = nums[j], nums[i]

================================================================
以上为个人学习笔记总结,供学习参考交流,未经允许禁止转载或商用。

  • 题目全部出自LeetCode官网:https://leetcode-cn.com/
  • 题目分类参考这篇文章:https://github.com/CyC2018/CS-Notes/blob/master/notes/Leetcode%20%E9%A2%98%E8%A7%A3%20-%20%E7%9B%AE%E5%BD%95.md

你可能感兴趣的:(LeetCode刷题笔记(算法思想 一))