代码随想录算法训练营第二十九天 | Leetcode随机抽题检测

news2025/1/12 22:49:29

Leetcode随机抽题检测

  • 160 相交链表
    • 未看解答自己编写的青春版
    • 重点
    • 题解的代码
    • 206 反转链表
  • 一段用于复制的标题
    • 未看解答自己编写的青春版
    • 重点
    • 题解的代码
    • 日后再次复习重新写
  • 234 回文链表
    • 未看解答自己编写的青春版
    • 重点
      • 综上,利用快慢指针找寻链表中间,就按加入虚拟头的方法写。
    • 题解的代码
    • 日后再次复习重新写
  • 141 环形链表
    • 未看解答自己编写的青春版
    • 重点
    • 题解的代码
    • 日后再次复习重新写
  • 142 环形链表 II
    • 未看解答自己编写的青春版
    • 重点
    • 题解的代码
    • 日后再次复习重新写
  • 21 合并两个有序链表
    • 未看解答自己编写的青春版
    • 重点
    • 题解的代码
    • 日后再次复习重新写
  • 2 两数相加
    • 未看解答自己编写的青春版
    • 重点
    • 题解的代码
    • 日后再次复习重新写
  • 19 删除链表的倒数第 N 个结点
    • 未看解答自己编写的青春版
    • 重点
      • 上面的代码其实有些冗余了,这道题只要能想到加入虚拟头,上面说的两种特殊情况都能迎刃而解,用一般的示例来考虑就能通过。卡哥的代码中,是先让 fast 移动 n+1 步,我的是先移动 n 步,都可以,不同的对用体现在while判断条件上。
    • 题解的代码
    • 日后再次复习重新写
  • 24 两两交换链表中的节点
    • 未看解答自己编写的青春版
    • 重点
    • 题解的代码
    • 日后再次复习重新写
  • 25 K 个一组翻转链表
    • 未看解答自己编写的青春版
    • 重点
      • 上面那个评论的方法真厉害,我悟了,第一次接触这种方式来翻转链表的,但是出人意料的很适合这道题!之前做的反转链表的方法,不能直接用到本题上,想使用就要像我那样去写,去谨慎的赋值。这种翻转的思想是:一次交换两个节点,且保持 pre 不变。例如:0--1--2--3--4,pre在0处,不参与翻转,cur在1处。那么第一次操作,是:0--2--1--3--4,此时pre还在0,cur也还在1,继续:0--3--2--1--4,继续:0--4--3--2--1,它是每次都将后面的一个值,插入到pre的后面,pre不移动,cur也不移动,但是cur的相对位置,会随着链表的更改而移动。
    • 题解的代码
    • 日后再次复习重新写
  • 138 复制带随机指针的链表
    • 未看解答自己编写的青春版
    • 重点
      • 通过对上面两种方法的学习,发现了这道题的本质,就是在利用next建立新链表的时候,利用一个map,保存好原节点和新节点的对应关系就好了!本质上是考哈希!
      • 另一种非常牛的方法,随机指针复制+拆分
      • 这题太牛了,一定要着重复习后面,这种思想第一次接触。
    • 题解的代码
    • 日后再次复习重新写
  • 148 排序链表
    • 未看解答自己编写的青春版
    • 重点,这道题太重要了太重要了,一定要多次复习。
      • 归并排序,是对链表排序最好的方法
      • 一定要好好学习归并排序,这是可以作为模板代码来学习的!本质上就是两个操作,merge 和 cut 。从网上搜到的很多归并排序都是用递归写的,不过本题使用循环来写,觉得这种写法很值得学习!
      • 归并排序的递归和循环写法,都值得学习!
      • 快速排序版本:是交换节点的,并非只交换数值。(暂时没看)
    • 题解的代码
    • 日后再次复习重新写
  • 23 合并 K 个升序链表
    • 未看解答自己编写的青春版
    • 重点
    • 题解的代码
    • 日后再次复习重新写
  • 146 LRU 缓存
    • 未看解答自己编写的青春版
    • 重点
      • 这道双向链表的题目,真的学习了,后面一定要多复习多重写这道题!
    • 题解的代码
    • 日后再次复习重新写
  • 94 二叉树的中序遍历
    • 未看解答自己编写的青春版
    • 重点
    • 题解的代码
    • 日后再次复习重新写
  • 104 二叉树的最大深度
    • 未看解答自己编写的青春版
    • 重点
    • 题解的代码
    • 日后再次复习重新写
  • 226 翻转二叉树
    • 未看解答自己编写的青春版
    • 重点
    • 题解的代码
    • 日后再次复习重新写
  • 101 对称二叉树
    • 未看解答自己编写的青春版
    • 重点
    • 题解的代码
    • 日后再次复习重新写
  • 543 二叉树的直径
    • 未看解答自己编写的青春版
    • 重点
    • 题解的代码
    • 日后再次复习重新写
  • 102 二叉树的层序遍历
    • 未看解答自己编写的青春版
    • 重点
    • 题解的代码
    • 日后再次复习重新写
  • 108 将有序数组转换为二叉搜索树
    • 未看解答自己编写的青春版
    • 重点
    • 题解的代码
    • 日后再次复习重新写
  • 98 验证二叉搜索树
    • 未看解答自己编写的青春版
    • 重点
    • 题解的代码
    • 日后再次复习重新写
  • 230 二叉搜索树中第K小的元素
    • 未看解答自己编写的青春版
    • 重点
    • 题解的代码
    • 日后再次复习重新写
  • 199 二叉树的右视图
    • 未看解答自己编写的青春版
    • 重点
    • 题解的代码
    • 日后再次复习重新写
  • 一段用于复制的标题
    • 未看解答自己编写的青春版
    • 重点
      • 额外空间复杂度为O(1)的思想!这道题我被题目骗了,题目说顺序应该是前序遍历,我就想着递归必须用前序遍历,但是前序遍历的问题是:会提前修改掉后面要进入递归的值。那么,链表,从上到下连接可以,我从下到上,反向连接,也可以啊!而这一点,利用递归的回溯特性,可以很自然地做到。假如 last 承载了下一层递归的返回值,而本层递归的root为last的上一个,只需要赋值:root.right=last,就可以了!
      • 所以后序遍历!后序遍历,也就不会有,还未进入递归,值就被修改的问题!这题的思路太妙了!
      • 思路很妙,可以多写多复习。
    • 题解的代码
    • 日后再次复习重新写
  • 105 从前序与中序遍历序列构造二叉树
    • 未看解答自己编写的青春版
      • 厉害!第一次自己尝试,写出了传入左右索引版的代码,在空间占用上,要比上面的代码低不少。
    • 重点
    • 题解的代码
    • 日后再次复习重新写
  • 437 路径总和 III
    • 未看解答自己编写的青春版
    • 重点
      • 没来得及体会,日后再看,重点学习这道题。
    • 题解的代码
    • 日后再次复习重新写
  • 236 二叉树的最近公共祖先
    • 未看解答自己编写的青春版
    • 重点
    • 题解的代码
    • 日后再次复习重新写
  • 124 二叉树中的最大路径和
    • 未看解答自己编写的青春版
    • 重点
    • 题解的代码
    • 日后再次复习重新写

160 相交链表

未看解答自己编写的青春版

class Solution:
    def getIntersectionNode(self, headA: ListNode, headB: ListNode) -> Optional[ListNode]:
        curA = headA
        curB = headB
        countA = 0
        countB = 0
        while curA :
            countA += 1
            curA = curA.next
        while curB :
            countB += 1
            curB = curB.next
        
        if countA < countB :
            countA,countB = countB,countA
            headA,headB = headB,headA
        diff = countA - countB
        while diff > 0 :
            headA = headA.next
            diff -= 1
        while headA :
            if headA == headB :
                return headB
            else :
                headA = headA.next
                headB = headB.next
        return None
        

重点

过。

题解的代码

206 反转链表

一段用于复制的标题

未看解答自己编写的青春版

class Solution:
    def reverseList(self, head: Optional[ListNode]) -> Optional[ListNode]:
        pre = None
        cur = head
        while cur :
            temp = cur.next
            cur.next = pre
            pre = cur
            cur = temp
        return pre

重点

过。

题解的代码

日后再次复习重新写

234 回文链表

未看解答自己编写的青春版

class Solution:
    def isPalindrome(self, head: Optional[ListNode]) -> bool:
        dummy_head = ListNode(0,head)
        slow = fast = dummy_head
        # 这里要注意,快慢指针找中间,要加虚拟头才是对的,不加虚拟头不对!
        # 自己举例一下就发现了。
        while fast and fast.next :
            slow = slow.next
            fast = fast.next.next
        cur = slow.next
        slow.next = None
        pre = None
        while cur :
            temp = cur.next
            cur.next = pre
            pre = cur
            cur = temp
        while pre and head :
            if pre.val != head.val :
                return False
            pre = pre.next
            head = head.next
        return True

重点

通过编写这道题的代码,发现了一个很重要的点,如果要用快慢指针法,寻找链表中点,最严格的方式是:要加入虚拟头!

加入虚拟头的代码:

		dummy_head = ListNode(0,head)
        slow = fast = dummy_head
        while fast and fast.next :
            slow = slow.next
            fast = fast.next.next
        cur = slow.next
        slow.next = None # 切断操作

不加入虚拟头的代码:

		slow = fast = head
        while fast and fast.next :
            slow = slow.next
            fast = fast.next.next
        cur = slow.next
        slow.next = None # 切断操作

下面用A代表慢指针,B代表快指针。

比如:1-2-3-4 ,不加入虚拟头时,一开始A,B均在 1 ,移动一步,A在2,B在3,此时还要继续移动,while不退出,继续移动,A在3,B为None,while退出,此时让A.next = None,做切断操作,这明显是错误的!应该是从 2 切断!加入虚拟头后,就是正确的结果。

加入虚拟头,可以让偶数个数的链表切割正确,奇数个数的链表切割结果不变,因为奇数个数的链表,正确切割结果,就是左边要包括中间节点,然后中间节点的next为None。

那么之前有一道题:143 重排链表,为什么卡哥的代码,也是快慢指针找中间,却没有加入虚拟头?
143 重排链表 卡哥的代码:

class Solution:
    def reorderList(self, head: Optional[ListNode]) -> None:
        """
        Do not return anything, modify head in-place instead.
        """
        fast = slow = head

        # find mid point which including (first) mid point into the first half linked list
        while fast and fast.next:
            fast = fast.next.next
            slow = slow.next
        # 下面两句代码别忘了,一个是取右半边,这样取的右半边一定是短的那一方
        # 这样连接也符合题目要求
        right = slow.next # 获取后半边的头
        slow.next = None # 切断!这句话很重要,不然就成环了
        node = None
        while right:
            temp = right.next
            right.next = node
            node = right
            right = temp
        head2 = node
        head1 = head
        while head1 and head2:
            temp1 = head1.next
            temp2 = head2.next
            head1.next = head2
            head2.next = temp1
            head1 = temp1
            head2 = temp2

在这里插入图片描述
因为这道题目的特殊性!可以看出,从2处切割,本题的节点串联逻辑是:1 – 4 – 2 – 3 。从3处切割,本题的节点串联逻辑是:1 – 4 – ( 2 3 ) ,( 2 3 ) 本来就在左子串中,无需改变位置!这是这道题的特殊性!对于偶数个数的链表,中间两个点,切不切割都一样!因为这两个点的顺序不需要颠倒。

综上,利用快慢指针找寻链表中间,就按加入虚拟头的方法写。

题解的代码

日后再次复习重新写

141 环形链表

未看解答自己编写的青春版

class Solution:
    def hasCycle(self, head: Optional[ListNode]) -> bool:
        if head == None or head.next == None :
            return False
        slow = fast = head
        while fast and fast.next : 
            slow = slow.next
            fast = fast.next.next
            # 注意判断的位置,先移动再判断,不然因为初始化都是头结点,会立刻返回True
            if slow == fast :
                return True
        return False

重点

在使用快慢指针时,注意 if 逻辑判断的位置。

注意!注意!警告!警告!上面代码中的提前判断,不是必须的。可以不加。

class Solution:
    def hasCycle(self, head: Optional[ListNode]) -> bool:
        slow = fast = head
        while fast and fast.next : 
            slow = slow.next
            fast = fast.next.next
            # 注意判断的位置,先移动再判断,不然因为初始化都是头结点,会立刻返回True
            if slow == fast :
                return True
        return False

题解的代码

日后再次复习重新写

142 环形链表 II

未看解答自己编写的青春版

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

重点

过,这道题的理论基础,去复习,卡哥的解答。

环形链表 II

题解的代码

日后再次复习重新写

21 合并两个有序链表

未看解答自己编写的青春版

class Solution:
    def mergeTwoLists(self, list1: Optional[ListNode], list2: Optional[ListNode]) -> Optional[ListNode]:
        newhead = ListNode()
        res = newhead
        while list1 and list2 :
            if list1.val >= list2.val :
                newhead.next = list2
                list2 = list2.next     
            else :
                newhead.next = list1
                list1 = list1.next
            newhead = newhead.next
        if list1 == None and list2 != None :
            newhead.next = list2
        elif list1 != None and list2 == None :
            newhead.next = list1
        return res.next

重点

过。

题解的代码

日后再次复习重新写

2 两数相加

未看解答自己编写的青春版

class Solution:
    def addTwoNumbers(self, l1: Optional[ListNode], l2: Optional[ListNode]) -> Optional[ListNode]:
        reshead = ListNode()
        cur = reshead
        pre = 0
        while l1 or l2 :
            if l1 == None :
                temp = pre+l2.val
                if temp > 9 :
                    pre = 1
                    temp = temp % 10
                else :
                    pre = 0
                node = ListNode(temp)
                l2 = l2.next
            elif l2 == None :
                temp = pre+l1.val
                if temp > 9 :
                    pre = 1
                    temp = temp % 10
                else :
                    pre = 0
                node = ListNode(temp)
                l1 = l1.next
            else :
                temp = pre+l2.val+l1.val
                if temp > 9 :
                    pre = 1
                    temp = temp % 10
                else :
                    pre = 0
                node = ListNode(temp)
                l1 = l1.next
                l2 = l2.next
            cur.next = node
            cur = cur.next
        if pre == 1 :
            node = ListNode(1)
            cur.next = node
        return reshead.next

重点

过。

题解的代码

日后再次复习重新写

19 删除链表的倒数第 N 个结点

未看解答自己编写的青春版

class Solution:
    def removeNthFromEnd(self, head: Optional[ListNode], n: int) -> Optional[ListNode]:
        dummy_head = ListNode(0,head)
        slow = fast = dummy_head
        while n > 0 :
            fast = fast.next
            n -= 1
        while fast and fast.next :
            fast = fast.next
            slow = slow.next
        slow.next = slow.next.next
        return dummy_head.next

重点

这道题就明确两点:

1、加入虚拟头。

2、模拟特殊情况检验代码是否正确:删除的是最后一个节点,删除的是第一个节点。

上面的代码其实有些冗余了,这道题只要能想到加入虚拟头,上面说的两种特殊情况都能迎刃而解,用一般的示例来考虑就能通过。卡哥的代码中,是先让 fast 移动 n+1 步,我的是先移动 n 步,都可以,不同的对用体现在while判断条件上。

移动 n+1 步 :

class Solution:
    def removeNthFromEnd(self, head: Optional[ListNode], n: int) -> Optional[ListNode]:
        dummy_head = ListNode(0,head)
        slow = fast = dummy_head
        while n > -1 :
            fast = fast.next
            n -= 1
        while fast :
            fast = fast.next
            slow = slow.next
        slow.next = slow.next.next
        return dummy_head.next

移动 n 步 :

class Solution:
    def removeNthFromEnd(self, head: Optional[ListNode], n: int) -> Optional[ListNode]:
        dummy_head = ListNode(0,head)
        slow = fast = dummy_head
        while n > 0 :
            fast = fast.next
            n -= 1
        while fast.next :
            fast = fast.next
            slow = slow.next
        slow.next = slow.next.next
        return dummy_head.next

题解的代码

日后再次复习重新写

24 两两交换链表中的节点

未看解答自己编写的青春版

class Solution:
    def swapPairs(self, head: Optional[ListNode]) -> Optional[ListNode]:
        if head == None or head.next == None :
            return head
        dummy_head = ListNode(0,head)

        pre = dummy_head
        cur = head
        while cur and cur.next :
            temp1 = cur.next
            temp2 = cur.next.next
            pre.next = temp1
            temp1.next = cur
            cur.next = temp2
            pre = cur
            cur = cur.next
        return dummy_head.next

重点

理清楚while循环中的交换逻辑就好,逻辑弄不清楚,就多搞一个临时变量嘛,两个temp,逻辑不就非常清晰。

这道题为了交换方便,同样也需要用虚拟头。

题解的代码

日后再次复习重新写

25 K 个一组翻转链表

未看解答自己编写的青春版

哈哈哈,独立完成 hard 题 !不过看评论,很多人也做出来了,看来这道题很简单,也就是中等的实际难度。

耗时上也还行,40% 左右。

class Solution:
    def reverseKGroup(self, head: Optional[ListNode], k: int) -> Optional[ListNode]:

        dummy_head = ListNode(0,head)
        pre = dummy_head
        begin = end = dummy_head.next
        count = 1
        while end :
            if count < k :
                end = end.next
                count += 1
            else :
                temp = end.next
                a,b = self.reverse(begin,end)
                pre.next = b
                pre = a
                a.next = temp
                begin = end = temp
                count = 1
        return dummy_head.next

    def reverse(self,head,end):
        end.next = None
        pre = None
        cur = head
        while cur :
            temp = cur.next
            cur.next = pre
            pre = cur
            cur = temp
        return head,end

重点

评论中的一个解答:

class Solution {
    public ListNode reverseKGroup(ListNode head, int k) {
        ListNode dummy = new ListNode(0), prev = dummy, curr = head, temp;
        dummy.next = head;
        int length = 0;
        while(head != null) {
            length++;
            head = head.next;
        }
        head = dummy.next;
        for(int i = 0; i < length / k; i++) {
            for(int j = 0; j < k - 1; j++) {
                temp = curr.next;
                curr.next = temp.next;
                temp.next = prev.next;
                prev.next = temp;
            }
            prev = curr;
            curr = prev.next;
        }
        return dummy.next;
    }
}

上面那个评论的方法真厉害,我悟了,第一次接触这种方式来翻转链表的,但是出人意料的很适合这道题!之前做的反转链表的方法,不能直接用到本题上,想使用就要像我那样去写,去谨慎的赋值。这种翻转的思想是:一次交换两个节点,且保持 pre 不变。例如:0–1–2–3–4,pre在0处,不参与翻转,cur在1处。那么第一次操作,是:0–2–1–3–4,此时pre还在0,cur也还在1,继续:0–3–2–1–4,继续:0–4–3–2–1,它是每次都将后面的一个值,插入到pre的后面,pre不移动,cur也不移动,但是cur的相对位置,会随着链表的更改而移动。

想清楚,节点的移动逻辑,再谨慎地编写每次循环中的逻辑就可以了。

class Solution:
    def reverseKGroup(self, head: Optional[ListNode], k: int) -> Optional[ListNode]:

        dummy_head = ListNode(0,head)
        pre = dummy_head
        cur = dummy_head.next
        aa = dummy_head.next
        count = 0
        while aa :
            count += 1
            aa = aa.next
        for i in range(count//k):
            for j in range(k-1):
                # 这里的四个赋值,顺序非常有讲究!
                temp = cur.next
                cur.next = temp.next
                temp.next = pre.next
                pre.next = temp
                  
            pre = cur
            cur = cur.next
        return dummy_head.next

题解的代码

日后再次复习重新写

138 复制带随机指针的链表

未看解答自己编写的青春版

思想很朴素,每次都从头搜索,random节点,所以耗时也只打败了6%。去学习一下评论区。

class Solution:
    def copyRandomList(self, head: 'Optional[Node]') -> 'Optional[Node]':
        reshead = Node(0)
        copycur = reshead
        cur = head
        while cur :
            node = Node(cur.val)
            copycur.next = node
            copycur = copycur.next
            cur = cur.next

        copycur = reshead.next
        cur = head
        while cur :
            if cur.random == None :
                copycur.random == None  
            else :
                temp_org = head
                temp_copy = reshead.next

                while temp_org != cur.random :
                    temp_org = temp_org.next
                    temp_copy = temp_copy.next
                copycur.random = temp_copy
            cur = cur.next
            copycur = copycur.next
        return reshead.next

重点

评论中的一个,递归的解法:很有意思。当然这种方法相当于记录了该链表的所有节点,空间复杂度是O(n),虽然是递归,但是这道题的时间复杂度也是O(n)。

class Solution:
    def copyRandomList(self, head):
        def copyNode(node, res):
            if not node: return None
            if node in res: return res[node]
            copy = Node(node.val, None, None)
            res[node] = copy
            copy.next = copyNode(node.next, res)
            copy.random = copyNode(node.random, res)
            return copy

        return copyNode(head, {})

另一种解法,哈希,利用字典去记录,原节点和新节点的映射关系。使用hash存储原结点和克隆结点的映射关系,通过映射关系处理克隆结点的random指针。

时间复杂度和空间复杂度,均为O(n)

class Solution {
    public Node copyRandomList(Node head) {
        if(head == null){
            return head;
        }
        // map方法,空间复杂度O(n)
        Node node = head;
        // 使用hash表存储旧结点和新结点的映射
        Map<Node,Node> map = new HashMap<>();
        while(node != null){
            Node clone = new Node(node.val,null,null);
            map.put(node,clone);
            node = node.next;
        }
        node = head;
        while(node != null){
            map.get(node).next = map.get(node.next);
            map.get(node).random = map.get(node.random);
            node = node.next;
        }
        return map.get(head);
    }
}

通过对上面两种方法的学习,发现了这道题的本质,就是在利用next建立新链表的时候,利用一个map,保存好原节点和新节点的对应关系就好了!本质上是考哈希!

另一种非常牛的方法,随机指针复制+拆分

原地处理,将克隆结点放在原结点后面,在原链表上处理克隆结点的random指针,最后分离两个链表,空间复杂度O(1)。

class Solution {
    public Node copyRandomList(Node head) {
        if(head == null){
            return head;
        }
        // 空间复杂度O(1),将克隆结点放在原结点后面
        Node node = head;
        // 1->2->3  ==>  1->1'->2->2'->3->3'
        while(node != null){
            Node clone = new Node(node.val,node.next,null);
            Node temp = node.next;
            node.next = clone;
            node = temp;
        }
        // 处理random指针
        node = head;
        while(node != null){
            // !!
            node.next.random = node.random == null ? null : node.random.next;
            node = node.next.next;
        }
        // 还原原始链表,即分离原链表和克隆链表
        node = head;
        Node cloneHead = head.next;
        while(node.next != null){
            Node temp = node.next;
            node.next = node.next.next;
            node = temp;
        }
        return cloneHead;
    }
}

一篇有助于理解的题解链接:

清楚的题解

这题太牛了,一定要着重复习后面,这种思想第一次接触。

题解的代码

日后再次复习重新写

哈希方法复写:

class Solution:
    def copyRandomList(self, head):
        reshead = Node(0)
        copycur = reshead
        cur = head
        table = {}
        while cur :
            node = Node(cur.val)
            copycur.next = node
            copycur = copycur.next
            table[cur] = copycur
            cur = cur.next

        copycur = reshead.next
        cur = head
        while cur :
            if cur.random == None :
                copycur.random == None  
            else :
                copycur.random = table[cur.random]
            cur = cur.next
            copycur = copycur.next
            
        return reshead.next

148 排序链表

未看解答自己编写的青春版

先遍历得到数组,再排序,用一个字典存储:数组排序后的下标(key)和链表中的节点(value)的映射,然后按照key,从小到大去索引原节点,来构造新链表。这是时间复杂度为O(n logn),空间复杂度为O(n)的做法。

速度很快,打败95%,但是空间上只打败了6%

class Solution:
    def sortList(self, head: Optional[ListNode]) -> Optional[ListNode]:
        table = {}
        nums = []
        cur = head
        while cur :
            # 处理值相同的情况
            if cur.val in table :
                # 这里要直接 append, 不能采用赋值操作:table[cur.val] = table[cur.val].append(cur)
                # 因为 table[cur.val].append(cur) 的返回值是None
                # 下面的 pop 操作同理
                table[cur.val].append(cur)
            else :
                table[cur.val] = [cur]
            nums.append(cur.val)
            cur = cur.next
        nums.sort()
        reshead = ListNode()
        cur = reshead
        
        for i in nums:
            # 注意这里,要从最末尾开始取,因为pop就是丢掉末尾的
            node = table[i][-1]  
            table[i].pop()
            cur.next = node
            cur = cur.next
        cur.next = None
        return reshead.next

要使用O(1)的空间复杂度,必然是对链表直接进行排序操作,对链表的排序还从来没写过,采用哪种方式好也不清楚,冒泡肯定不行,是O(n^2),看题解说是,归并排序,归并排序是最适合链表这种数据结构的排序方式。快速排序也可以。

重点,这道题太重要了太重要了,一定要多次复习。

归并排序,是对链表排序最好的方法

在这里插入图片描述
伪代码:

current = dummy.next;
tail = dummy;
for (step = 1; step < length; step *= 2) {
	while (current) {
		// left->@->@->@->@->@->@->null
		left = current;

		// left->@->@->null   right->@->@->@->@->null
		right = cut(current, step); // 将 current 切掉前 step 个头切下来。

		// left->@->@->null   right->@->@->null   current->@->@->null
		current = cut(right, step); // 将 right 切掉前 step 个头切下来。
		
		// dummy.next -> @->@->@->@->null,最后一个节点是 tail,始终记录
		//                        ^
		//                        tail
		tail.next = merge(left, right);
		while (tail->next) tail = tail->next; // 保持 tail 为尾部
	}
}

正式代码:好像是C ?

class Solution {
public:
    ListNode* sortList(ListNode* head) {
        ListNode dummyHead(0);
        dummyHead.next = head;
        auto p = head;
        int length = 0;
        while (p) {
            ++length;
            p = p->next;
        }
        
        for (int size = 1; size < length; size <<= 1) {
            auto cur = dummyHead.next;
            auto tail = &dummyHead;
            
            while (cur) {
                auto left = cur;
                auto right = cut(left, size); // left->@->@ right->@->@->@...
                cur = cut(right, size); // left->@->@ right->@->@  cur->@->...
                
                tail->next = merge(left, right);
                while (tail->next) {
                    tail = tail->next;
                }
            }
        }
        return dummyHead.next;
    }
    
    ListNode* cut(ListNode* head, int n) {
        auto p = head;
        while (--n && p) {
            p = p->next;
        }
        
        if (!p) return nullptr;
        
        auto next = p->next;
        p->next = nullptr;
        return next;
    }
    
    ListNode* merge(ListNode* l1, ListNode* l2) {
        ListNode dummyHead(0);
        auto p = &dummyHead;
        while (l1 && l2) {
            if (l1->val < l2->val) {
                p->next = l1;
                p = l1;
                l1 = l1->next;       
            } else {
                p->next = l2;
                p = l2;
                l2 = l2->next;
            }
        }
        p->next = l1 ? l1 : l2;
        return dummyHead.next;
    }
};

一定要好好学习归并排序,这是可以作为模板代码来学习的!本质上就是两个操作,merge 和 cut 。从网上搜到的很多归并排序都是用递归写的,不过本题使用循环来写,觉得这种写法很值得学习!

归并排序的递归和循环写法,都值得学习!

快速排序版本:是交换节点的,并非只交换数值。(暂时没看)

class Solution {
public ListNode sortList(ListNode head) {
        if(head==null||head.next==null) return head;
        // 没有条件,创造条件。自己添加头节点,最后返回时去掉即可。
        ListNode newHead=new ListNode(-1);
        newHead.next=head;
        return quickSort(newHead,null);
    }
    // 带头结点的链表快速排序
    private ListNode quickSort(ListNode head,ListNode end){
        if (head==end||head.next==end||head.next.next==end) return head;
        // 将小于划分点的值存储在临时链表中
        ListNode tmpHead=new ListNode(-1);
        // partition为划分点,p为链表指针,tp为临时链表指针
        ListNode partition=head.next,p=partition,tp=tmpHead;
        // 将小于划分点的结点放到临时链表中
        while (p.next!=end){
            if (p.next.val<partition.val){
                tp.next=p.next;
                tp=tp.next;
                p.next=p.next.next;
            }else {
                p=p.next;
            }
        }
        // 合并临时链表和原链表,将原链表接到临时链表后面即可
        tp.next=head.next;
        // 将临时链表插回原链表,注意是插回!(不做这一步在对右半部分处理时就断链了)
        head.next=tmpHead.next;
        quickSort(head,partition);
        quickSort(partition,end);
        // 题目要求不带头节点,返回结果时去除
        return head.next;
    }
}

题解的代码

日后再次复习重新写

自己复写的归并排序:好好理解!

class Solution:
    def sortList(self, head: Optional[ListNode]) -> Optional[ListNode]:
        dummy_head = ListNode(0,head)
        cur = head
        count = 0
        while cur :
            count += 1
            cur = cur.next
        step = 1
        while step < count :
            # 这里cur从头结点head开始也是必须的,而不能从虚拟头开始
            cur = dummy_head.next
            pre = dummy_head
            while cur :
                left = cur
                right = self.cut(left,step)

                cur = self.cut(right,step)
                pre.next = self.merge(left,right)
                while pre.next :
                    pre = pre.next
            step = step << 1
        return dummy_head.next
        

    def merge(self,head1,head2):
        dummy_head = ListNode()
        cur = dummy_head
        while head1 and head2 :
            if head1.val >= head2.val :
                cur.next = head2
                head2 = head2.next
                cur = cur.next
            else :
                cur.next = head1
                head1 = head1.next
                cur = cur.next
        if head1 == None :
            cur.next = head2
        else :
            cur.next = head1
        return dummy_head.next


    # 注意,cut操作是,返回切断后,后半部分的链表头
    def cut(self,head,n):
        cur = head
        # 注意这里,一定是n>1 ,因为left是从头结点head开始的
        # 那么如果step是1的话,指针应该不移动,这样才能仅cut掉当前节点
        while n > 1 and cur != None :
            cur = cur.next
            n -= 1
        if cur == None :
            return None
        res = cur.next
        # cut 操作要在最后结尾处截断,赋值为 None
        cur.next = None
        return res


23 合并 K 个升序链表

未看解答自己编写的青春版

有了上一题的铺垫,这道题就显得较为简单了,但是效率高不高就不清楚了。倒序归并排序,执行 n-1 次,因为是每次合并两个,然后 pop 出这两个,然后将结果 append 进去。

class Solution:
    def mergeKLists(self, lists: List[Optional[ListNode]]) -> Optional[ListNode]:
        n = len(lists) 
        if lists == []:
            return None
        temp = lists[0]
        while len(lists) > 1 :
            head1 = lists[-1]
            head2 = lists[-2]
            temp = self.merge(head1,head2)
            lists.pop()
            lists.pop()
            lists.append(temp)
        return temp





    def merge(self,head1,head2):
        dummy_head = ListNode()
        cur = dummy_head
        while head1 and head2 :
            if head1.val >= head2.val :
                cur.next = head2
                head2 = head2.next
                cur = cur.next
            else :
                cur.next = head1
                head1 = head1.next
                cur = cur.next
        if head1 == None :
            cur.next = head2
        else :
            cur.next = head1
        return dummy_head.next

重点

这题的评论区里,一堆妖魔鬼怪的方法,作为初学者的我,就找一个最朴素的思路吧,分治法。

可以学习一下,直接没怎么接触过?

其实就是递归,只不过之前的递归,在获得 L1 和 L2 处,要收获结果了,都是一个类似于:加和,append的操作,本题变成了一个两个链表的merge函数。

class Solution:
    def mergeKLists(self, lists: List[ListNode]) -> ListNode:

        n = len(lists)

        def merge(left, right):
            if left > right:
                return
            if left == right:
                return lists[left]
            mid = (left + right) // 2
            l1 = merge(left, mid)
            l2 = merge(mid + 1, right)
            return mergeTwoLists(l1, l2)

        def mergeTwoLists(l1, l2):
            if not l1 or not l2:
                return l1 or l2
            if l1.val < l2.val:
                l1.next = mergeTwoLists(l1.next, l2)
                return l1
            else:
                l2.next = mergeTwoLists(l1, l2.next)
                return l2

        return merge(0, n - 1)

题解的代码

日后再次复习重新写

分治法复写:将原本代码中,归并两个有序链表的 mergeTwoLists 函数的递归写法,改为了一般的循环写法,这样看上去就更好理解一些了,本质上就是递归!

class Solution:
    def mergeKLists(self, lists: List[Optional[ListNode]]) -> Optional[ListNode]:
      
        n = len(lists)

        def merge(left, right):
            if left > right:
                return
            if left == right:
                return lists[left]
            mid = (left + right) // 2
            l1 = merge(left, mid)
            l2 = merge(mid + 1, right)
            return self.mergeTwoLists(l1, l2)

        return merge(0, n - 1)



    def mergeTwoLists(self,head1,head2):
        dummy_head = ListNode()
        cur = dummy_head
        while head1 and head2 :
            if head1.val >= head2.val :
                cur.next = head2
                head2 = head2.next
                cur = cur.next
            else :
                cur.next = head1
                head1 = head1.next
                cur = cur.next
        if head1 == None :
            cur.next = head2
        else :
            cur.next = head1
        return dummy_head.next

146 LRU 缓存

未看解答自己编写的青春版

没见过这种类型的题,也不知道应该用什么数据结构。

重点

首先要明确本题的两个要点。

1、LRU 的功能可以使用双向链表实现,访问到的节点移动到头部,超出容量的从尾部删除。

2、要实现O(1)得使用HaspMap,里面储存 key 与 链表节点即可,这样可以快速定位节点,然后删除它,将它移动到链表头部。

这道双向链表的题目,真的学习了,后面一定要多复习多重写这道题!

题解的代码

class ListNode:
    def __init__(self, key=None, value=None):
        self.key = key
        self.value = value
        self.prev = None
        self.next = None


class LRUCache:
    def __init__(self, capacity: int):
        self.capacity = capacity
        self.hashmap = {}
        # 新建两个节点 head 和 tail
        self.head = ListNode()
        self.tail = ListNode()
        # 初始化链表为 head <-> tail
        self.head.next = self.tail
        self.tail.prev = self.head

    # 因为get与put操作都可能需要将双向链表中的某个节点移到末尾,所以定义一个方法
    def move_node_to_tail(self, key):
            # 先将哈希表key指向的节点拎出来,为了简洁起名node
            #      hashmap[key]                               hashmap[key]
            #           |                                          |
            #           V              -->                         V
            # prev <-> node <-> next         pre <-> next   ...   node
            node = self.hashmap[key]
            node.prev.next = node.next
            node.next.prev = node.prev
            # 之后将node插入到尾节点前
            #                 hashmap[key]                 hashmap[key]
            #                      |                            |
            #                      V        -->                 V
            # prev <-> tail  ...  node                prev <-> node <-> tail
            node.prev = self.tail.prev
            node.next = self.tail
            self.tail.prev.next = node
            self.tail.prev = node

    def get(self, key: int) -> int:
        if key in self.hashmap:
            # 如果已经在链表中了久把它移到末尾(变成最新访问的)
            self.move_node_to_tail(key)
        res = self.hashmap.get(key, -1)
        if res == -1:
            return res
        else:
            return res.value

    def put(self, key: int, value: int) -> None:
        if key in self.hashmap:
            # 如果key本身已经在哈希表中了就不需要在链表中加入新的节点
            # 但是需要更新字典该值对应节点的value
            self.hashmap[key].value = value
            # 之后将该节点移到末尾
            self.move_node_to_tail(key)
        else:
            if len(self.hashmap) == self.capacity:
                # 去掉哈希表对应项
                self.hashmap.pop(self.head.next.key)
                # 去掉最久没有被访问过的节点,即头节点之后的节点
                self.head.next = self.head.next.next
                self.head.next.prev = self.head
            # 如果不在的话就插入到尾节点前
            new = ListNode(key, value)
            self.hashmap[key] = new
            new.prev = self.tail.prev
            new.next = self.tail
            self.tail.prev.next = new
            self.tail.prev = new

日后再次复习重新写

94 二叉树的中序遍历

未看解答自己编写的青春版

class Solution:
    def inorderTraversal(self, root: Optional[TreeNode]) -> List[int]:
        if root == None :
            return []
        left = self.inorderTraversal(root.left)
        middle = root.val
        right = self.inorderTraversal(root.right)

        return left+[middle]+right
        

重点

过。本次刷题旨在把题做出来,不考虑,递归法怎么写,迭代法怎么写,其他方法有没有这种事情了。

题解的代码

日后再次复习重新写

104 二叉树的最大深度

未看解答自己编写的青春版

class Solution:
    def maxDepth(self, root: Optional[TreeNode]) -> int:
        if root == None :
            return 0
        left = self.maxDepth(root.left)
        right = self.maxDepth(root.right)
        return 1 + max(left,right)

重点

最大深度 = 根节点的最大高度,直接用高度的定义去递归,简单!

题解的代码

日后再次复习重新写

226 翻转二叉树

未看解答自己编写的青春版

正确可以AC的代码:

class Solution:
    def invertTree(self, root: Optional[TreeNode]) -> Optional[TreeNode]:
        if root == None :
            return None
        left = self.invertTree(root.left)
        right = self.invertTree(root.right)
        root.left , root.right = right,left
        return root

要引以为戒的错误代码:错误原因,在未完全递归完成之前,就改变了当前正在递归节点的左右子树。

class Solution:
    def invertTree(self, root: Optional[TreeNode]) -> Optional[TreeNode]:
        if root == None :
            return None
        root.left = self.invertTree(root.right)
        # 上一行代码的赋值操作,导致改变了root.left,下面的递归就不对了!
        root.right = self.invertTree(root.left)
        return root

重点

题解的代码

日后再次复习重新写

101 对称二叉树

未看解答自己编写的青春版

这道题没法在原函数上进行递归判断了,必须新建一个函数,因为对称的判断需要左右两棵子树的头结点。

class Solution:   
    def isSymmetric(self, root: Optional[TreeNode]) -> bool:
        if root == None :
            return True
        return self.judge_symmetric(root.left,root.right)

    def judge_symmetric(self,left,right):
        if left == None and right == None :
            return True
        elif left == None and right != None :
            return False
        elif left != None and right == None :
            return False
        else :
            flag1 = self.judge_symmetric(left.left,right.right)
            flag2 = self.judge_symmetric(left.right,right.left)
            if left.val == right.val :
                flag3 = True
            else :
                flag3 = False
            return flag1 and flag2 and flag3

怎么感觉我这次写的这个代码,有点复杂呢。

重点

之前的代码也差不多,可以通过更改顺序,稍微优化一下。

# Definition for a binary tree node.
# class TreeNode:
#     def __init__(self, val=0, left=None, right=None):
#         self.val = val
#         self.left = left
#         self.right = right
class Solution:  
    def isSymmetric(self, root: Optional[TreeNode]) -> bool:
        if root == None :
            return True
        return self.digui(root.left,root.right)
        
    def digui(self,p,q):
        if p == None and q == None :
            return True
        elif p == None and q != None :
            return False
        elif p != None and q == None :
            return False
        else :
            if p.val != q.val :
                return False
            else :
                left = self.digui(p.left,q.right)
                if left :
                    right = self.digui(p.right,q.left)
                return left and right
        

题解的代码

日后再次复习重新写

543 二叉树的直径

未看解答自己编写的青春版

嘿嘿,无敌。虽然这道题是简单题,但是我觉得这道题如何保存最长的路径,以及怎么处理每个节点的返回值,还蛮需要考虑清楚的。

class Solution:    
    def diameterOfBinaryTree(self, root: Optional[TreeNode]) -> int:
        if root == None :
            return 0
        self.res = 0
        self.track_road(root)
        return self.res

    def track_road(self,root):
        if root == None :
            return 0
        left = self.track_road(root.left)
        right = self.track_road(root.right)
        self.res = max(self.res,left+right)
        return 1 + max(left,right)

重点

我自认为重点有两个:

1、最长路径不一定经过根节点,比如根节点的左子树很深,而且是完全二叉树,而根节点的右子树只有一个节点,那么最长路径一定出现在左子树中。

2、本题要处理的值,和节点的返回值不一致,不是像之前做过的题目,所求结果就是根节点的返回值!在当前节点,最长路径是左右子树加起来,但是如果要返回到上一层,只能选一个max的返回!

题解的代码

日后再次复习重新写

102 二叉树的层序遍历

未看解答自己编写的青春版

from collections import deque
class Solution:
    def levelOrder(self, root: Optional[TreeNode]) -> List[List[int]]:
        if root == None :
            return []
        dq = deque()
        dq.append(root)
        res = []
        while dq :
            size = len(dq)
            level = []
            for i in range(size):
                node = dq.popleft()
                level.append(node.val)
                if node.left :
                    dq.append(node.left)
                if node.right :
                    dq.append(node.right)
            res.append(level)
        return res

重点

层序遍历,模板题。

题解的代码

日后再次复习重新写

108 将有序数组转换为二叉搜索树

未看解答自己编写的青春版

递归构造就行了。

class Solution:
    def sortedArrayToBST(self, nums: List[int]) -> Optional[TreeNode]:
        if nums == [] :
            return None
        n = len(nums)
        mid = n // 2
        node = TreeNode(nums[mid])
        node.left = self.sortedArrayToBST(nums[0:mid])
        node.right = self.sortedArrayToBST(nums[mid+1:])
        return node

重点

题解的代码

日后再次复习重新写

98 验证二叉搜索树

未看解答自己编写的青春版

中序遍历:左中右,所有的中间节点处理逻辑都要放在中间。

class Solution:
    def __init__(self):
        self.pre = None
    def isValidBST(self, root: Optional[TreeNode]) -> bool:
        if root == None :
            return True
        left = self.isValidBST(root.left)
        # 中序遍历:左中右,所有的中间节点处理逻辑都要放在中间
        if self.pre :
            if self.pre.val >= root.val :
                return False
        # 这句话一定要放在中间
        self.pre = root
        right = self.isValidBST(root.right)       
        return left and right

赋值语句 ( self.pre = root ) ,位置错误,导致的错误代码:

class Solution:
    def __init__(self):
        self.pre = None
    def isValidBST(self, root: Optional[TreeNode]) -> bool:
        if root == None :
            return True
        left = self.isValidBST(root.left)
        # 中序遍历:左中右,所有的中间节点处理逻辑都要放在中间
        if self.pre :
            if self.pre.val >= root.val :
                return False
        right = self.isValidBST(root.right)    
        self.pre = root
        return left and right

中序遍历迭代法:

class Solution:
    def isValidBST(self, root: Optional[TreeNode]) -> bool:
        stack = []
        cur = root
        pre = None
        while stack or cur :
            if cur :
                stack.append(cur)
                cur = cur.left
            else :
                node = stack.pop()
                if pre :
                    if pre.val >= node.val :
                        return False
                pre = node
                cur = node.right
        return True

上面从stack里 pop 出来的值,可以直接用 cur 承接,更顺眼一些。

class Solution:
    def isValidBST(self, root: Optional[TreeNode]) -> bool:
        stack = []
        cur = root
        pre = None
        while stack or cur :
            if cur :
                stack.append(cur)
                cur = cur.left
            else :
                cur = stack.pop()
                if pre :
                    if pre.val >= cur.val :
                        return False
                pre = cur
                cur = cur.right
        return True

重点

题解的代码

日后再次复习重新写

230 二叉搜索树中第K小的元素

未看解答自己编写的青春版

中序遍历的迭代法模板,直接秒了,但是时间上只打败了7% ?

class Solution:
    def kthSmallest(self, root: Optional[TreeNode], k: int) -> int:
        count = 0
        stack = []
        cur = root
        while stack or cur :
            if cur :
                stack.append(cur)
                cur = cur.left
            else :
                cur = stack.pop()
                count += 1
                if count == k:
                    return cur.val
                cur = cur.right
        return 0

重点

从网上看了一些评论,也基本上和我的方法一致。

另一种思路:通过计算节点个数来找寻第K个数,查找左子树节点个数为 leftN , 如果 K<=leftN ,则所查找节点在左子树上,若 K=leftN+1 , 则所查找节点为根节点,若 K>leftN+1 , 则所查找节点在右子树上, 按照同样方法查找右子树第 K-leftN 个节点。

但是在时间上还是打败 7% , 其他解法怎么这么快的?

class Solution:
    def kthSmallest(self, root: Optional[TreeNode], k: int) -> int:
        
        left = self.count(root.left)
        if left + 1 == k:
            return root.val
        elif left >= k :
            return self.kthSmallest(root.left,k)
        else :
            return self.kthSmallest(root.right,k-left-1)

    def count(self,root):
        if root == None :
            return 0
        return self.count(root.left)+self.count(root.right)+1

题解的代码

力扣的示例代码:

class Solution:
    def kthSmallest(self, root: Optional[TreeNode], k: int) -> int:      
        res = []
        def dfs(node):
            if not node:
                return
            if len(res)>=k:
                return
            dfs(node.left)
            res.append(node.val)
            dfs(node.right)
        dfs(root)
        return res[k-1]

原来就是递归,储存所有遍历过的节点,当结果列表长度大于等于 k 时,返回。

看起来因为使用的是递归,所以应该是比我的中序遍历迭代法要少遍历一些节点。不过这个代码,有时候90%,有时候也是7%,不纠结这道题的耗时统计了!

日后再次复习重新写

199 二叉树的右视图

未看解答自己编写的青春版

又是层序遍历模板题。

from collections import deque
class Solution:
    def rightSideView(self, root: Optional[TreeNode]) -> List[int]:
        if root == None :
            return []
        dq = deque()
        dq.append(root)
        res = []
        while dq :
            size = len(dq)
            for i in range(size):
                node = dq.popleft()
                if node.left :
                    dq.append(node.left)
                if node.right:
                    dq.append(node.right)
            res.append(node.val)
        return res

重点

题解的代码

日后再次复习重新写

一段用于复制的标题

未看解答自己编写的青春版

一开始总想着用递归做,但是发现,在递归中,无法时刻保存当前需要赋值的节点,因为本题要求不能返回任意值,也就是要操作原节点,所以递归函数返回值,赋值,这些操作都是无效的,所以应该用前序遍历的迭代法。

class Solution:
    def flatten(self, root: Optional[TreeNode]) -> None:
        """
        Do not return anything, modify root in-place instead.
        """
        if root == None :
            return None
        stack = [root]
        cur = root
        while stack :
            node = stack.pop()
            if node == root :
                pass
            else :
                cur.right = node
                cur.left = None
                cur = node
            if node.right :
                stack.append(node.right)
            if node.left :
                stack.append(node.left)

但是上述是:额外空间复杂度为O(n),因为额外申请了一个堆栈。

重点

额外空间复杂度为O(1)的思想!这道题我被题目骗了,题目说顺序应该是前序遍历,我就想着递归必须用前序遍历,但是前序遍历的问题是:会提前修改掉后面要进入递归的值。那么,链表,从上到下连接可以,我从下到上,反向连接,也可以啊!而这一点,利用递归的回溯特性,可以很自然地做到。假如 last 承载了下一层递归的返回值,而本层递归的root为last的上一个,只需要赋值:root.right=last,就可以了!

所以后序遍历!后序遍历,也就不会有,还未进入递归,值就被修改的问题!这题的思路太妙了!

class Solution:
    def __init__(self):
        self.last = None
    def flatten(self, root: Optional[TreeNode]) -> None:
        """
        Do not return anything, modify root in-place instead.
        """
        if root == None :
            return None
        self.flatten(root.right)
        self.flatten(root.left)
        root.right = self.last
        root.left = None
        self.last = root

思路很妙,可以多写多复习。

题解的代码

日后再次复习重新写

105 从前序与中序遍历序列构造二叉树

未看解答自己编写的青春版

递归,数组切片方法:

class Solution:
    def buildTree(self, preorder: List[int], inorder: List[int]) -> Optional[TreeNode]:
        if preorder == []:
            return None
        middle = preorder[0]
        index = inorder.index(middle)
        
        node = TreeNode(middle)
        node.left = self.buildTree(preorder[1:index+1],inorder[:index])
        node.right = self.buildTree(preorder[index+1:],inorder[index+1:])
        return node

厉害!第一次自己尝试,写出了传入左右索引版的代码,在空间占用上,要比上面的代码低不少。

class Solution:
    def buildTree(self, preorder: List[int], inorder: List[int]) -> Optional[TreeNode]:

        def build(pleft,pright,ileft,iright):
            # 左闭右开,不包括right,所以才能有等于号
            if pleft >= pright :
                return None
            middle = preorder[pleft]
            index = inorder.index(middle)
            node = TreeNode(middle)
            # 举例: ileft = 3 , middle = 4 , 意味着左子树只有一个节点
            # 那么对于前序数组,起始位置 pleft+1 , 要想让左子树有一个点,区间应该为
            # [pleft+1,pleft+2) 注意循环不变量,左闭右开
            node.left = build(pleft+1,pleft+1+index-ileft,ileft,index)
            node.right = build(pleft+1+index-ileft,pright,index+1,iright)
      
            return node

        n = len(preorder)
        return build(0,n,0,n)

重点

过。可以体会下,传入左右区间的参数的版本,怎样能把每个区间都写对。

题解的代码

日后再次复习重新写

437 路径总和 III

未看解答自己编写的青春版

没做出来。

重点

主要是两种方法:双重递归 ; 前缀和。

力扣官方题解

没来得及体会,日后再看,重点学习这道题。

题解的代码

双重递归:

class Solution:
    def pathSum(self, root: TreeNode, targetSum: int) -> int:
        def rootSum(root, targetSum):
            if root is None:
                return 0

            ret = 0
            if root.val == targetSum:
                ret += 1

            ret += rootSum(root.left, targetSum - root.val)
            ret += rootSum(root.right, targetSum - root.val)
            return ret
        
        if root is None:
            return 0
            
        ret = rootSum(root, targetSum)
        ret += self.pathSum(root.left, targetSum)
        ret += self.pathSum(root.right, targetSum)
        return ret

前缀和:

class Solution:
    def pathSum(self, root: TreeNode, targetSum: int) -> int:
        prefix = collections.defaultdict(int)
        prefix[0] = 1

        def dfs(root, curr):
            if not root:
                return 0
            
            ret = 0
            curr += root.val
            ret += prefix[curr - targetSum]
            prefix[curr] += 1
            ret += dfs(root.left, curr)
            ret += dfs(root.right, curr)
            prefix[curr] -= 1

            return ret

        return dfs(root, 0)

日后再次复习重新写

236 二叉树的最近公共祖先

未看解答自己编写的青春版

搞懂,左右公共祖先,怎么从后序遍历中,层层回溯到根节点的。

class Solution:
    def lowestCommonAncestor(self, root: 'TreeNode', p: 'TreeNode', q: 'TreeNode') -> 'TreeNode':
        self.res = None
        self.digui(root,p,q)
        return self.res

    def digui(self,root,p,q):
        if root == None :
            return False
        if root.val == p.val or root.val == q.val :
            self.res = root
            return True               
        left = self.digui(root.left,p,q)
        right = self.digui(root.right,p,q)
        if left and right :
            self.res = root
        return left or right

重点

参考卡哥的解答。

二叉树的最近公共祖先

题目拓展:如果树是一个二叉搜索树呢?前序遍历树中节点的值就可以了,当节点值第一次出现在在区间
[ p.val , q.val ] 时,这个节点就是最近公共祖先,可以用反证法证明,再走一步就不符合条件了。

题解的代码

日后再次复习重新写

124 二叉树中的最大路径和

未看解答自己编写的青春版

小小 hard 。

本题只需要考虑清楚:对于在当前节点收获结果的逻辑:左和右,分别都有两个状态,取或不取,一共四种情况就好了;递归函数在返回时,同前面做过的一道题,返回的时候,没有左右都考虑的情况。

class Solution:
    def maxPathSum(self, root: Optional[TreeNode]) -> int:
        self.res = -inf
        self.digui(root)
        return self.res

    def digui(self,root):
        if root == None :
            return 0
        if root.left == None and root.right == None :
            self.res = max(self.res,root.val)
            return root.val
        left = self.digui(root.left)
        right = self.digui(root.right)
        # 只要想明白这里,左和右,分别都有两个状态,取或不取,一共四种情况就好了
        self.res = max(self.res,left+right+root.val,root.val,right+root.val,left+root.val)
        # 同前面做过的一道题,返回的时候,没有左右都考虑的情况
        return max(left,right,0)+root.val

重点

题解的代码

日后再次复习重新写

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/823129.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【C++】总结9

文章目录 C从源代码到可执行程序经过什么步骤静态链接和动态链接类的对象存储空间C的内存分区内存池在成员函数中调用delete this会出现什么问题&#xff1f;如果在类的析构函数中调用delete this&#xff0c;会发生什么&#xff1f; C从源代码到可执行程序经过什么步骤 预处理…

Tomcat 创建https

打开CMD,按下列输入 keytool -genkeypair -alias www.bo.org -keyalg RSA -keystore d:\ambition.keystore -storetype pkcs12 输入密钥库口令:123456 再次输入新口令:123456 您的名字与姓氏是什么? [Unknown]: www.ambition.com 您的组织单位名称是什么? [Unknown…

Qt 编译 Android 项目,输出乱码

乱码如下&#xff1a; :-1: error: 娉 C:\Qt\6.5.0\android_arm64_v8a\src\android\java\src\org\qtproject\qt\android\bindings\QtActivity.java浣跨敤鎴栬鐩栦簡宸茶繃鏃剁殑 API銆 娉 鏈夊叧璇︾粏淇℃伅, 璇蜂娇鐢-Xlint:deprecation 閲嶆柊缂栬瘧銆 正确的应该是&#…

qemu kvm 新建虚拟机

开始菜单打开虚拟机管理器

HDFS集群滚动升级以及回滚相关

HDFS集群滚动升级以及回滚相关 介绍不停机滚动升级非联邦HA集群联邦HA集群 停机升级--非HA集群HDFS集群降级和回滚异同点共同点不同点 HA集群降级&#xff08;downgrade&#xff09;注意事项 集群回滚操作 介绍 在hadoop v2中&#xff0c;HDFS支持namenode高可用&#xff08;H…

neo4j使用中的常见问题

1Spring Boot NEO The client is unauthorized due to authentication failure 解决方法&#xff1a;找到你安装neo4j的路径下的conf文件夹&#xff0c;找到neo4j.conf #dbms.security.auth_enabledfalse将前面的注释#去掉&#xff0c;然后重启neo4j&#xff0c;在重启项目即…

el-cascader级联选择器加载远程数据、默认开始加载固定条、可以根据搜索加载远程数据。

加载用户列表分页请求、默认请求20条数据。想添加远程搜索用户功能。原有的方法filter-method不能监听到输入清空数据的时候。这样搜索完无法返回默认的20条数据。直接监听级联选择的v-model绑定的值是无法检测到用户自己输入的。 解决思路&#xff1a; el-cascader 没有提供…

屏蔽托盘右键菜单

最近有个需求需要屏蔽托盘图标的右下角菜单项&#xff1a; 经过Apimonitor进行hook Explorer进程&#xff0c;发现弹出菜单是通过explorer调用InserMenuItem函数来实现的。通过注入explorer并挂钩InserMenuItemW函数&#xff0c;并屏蔽自己想要屏蔽的菜单项&#xff1a; &#…

2023-08-01 LeetCode每日一题(英雄的力量)

2023-08-01每日一题 一、题目编号 2681. 英雄的力量二、题目链接 点击跳转到题目位置 三、题目描述 给你一个下标从 0 开始的整数数组 nums &#xff0c;它表示英雄的能力值。如果我们选出一部分英雄&#xff0c;这组英雄的 力量 定义为&#xff1a; i0 &#xff0c;i1 &…

【ARM Coresight 系列文章 2.5 - Coresight 寄存器:PIDR0-PIDR7,CIDR0-CIDR3 介绍】

文章目录 1.1 JEDEC 与 JEP1061.2 PIDR0-PIDR7(peripheral identification registers)1.2 CIDR0-CIDR3(Component Identification Registers) 1.1 JEDEC 与 JEP106 JEDEC和JEP106都是来自美国电子工业联合会&#xff08;JEDEC&#xff0c;Joint Electron Device Engineering C…

Kafka3.0.0版本——Broker(总体工作流程)

目录 一、Kafka中Broker总体工作流程图解二、Kafka中Broker总体工作流程步骤解析 一、Kafka中Broker总体工作流程图解 总体工作流程图解 二、Kafka中Broker总体工作流程步骤解析 1、broker启动后在zk中注册&#xff0c;如下图所示&#xff1a; 2、controller谁先注册&…

Java面向对象之UML类图

UML类图 表示 public 类型&#xff0c; - 表示 private 类型&#xff0c;#表示protected类型方法的写法&#xff1a;方法的类型(、-) 方法名(参数名&#xff1a; 参数类型)&#xff1a;返回值类型

Windows下安装Spark(亲测成功安装)

Windows下安装Spark 一、Spark安装前提1.1、JDK安装&#xff08;version&#xff1a;1.8&#xff09;1.1.1、JDK官网下载1.1.2、JDK网盘下载1.1.3、JDK安装 1.2、Scala安装&#xff08;version&#xff1a;2.11.12&#xff09;1.2.1、Scala官网下载1.2.2、Scala网盘下载1.2.3、…

3DEXPERIENCE用户角色 | Structural Mechanics Engineer 结构力学工程师

真实条件下实施复杂的线性和非线性分析 直观验证设计并更快地做出产品决策 Structural Mechanics Engineer 在基于云的 3DEXPERIENCE 平台上构建&#xff0c;您可对产品行为执行结构线性和非线性静态、低速和高速动态和热仿真。具备材料校准功能&#xff0c;有助于确保材料行为…

wine意大利红酒数据标准化案例

1.数据和环境准备 将通过意大利红酒的部分数据&#xff0c;调用scikit-learn包&#xff08;sklearn&#xff09;分别实现0-1标准化和z-score标准化&#xff0c;总结学习这两种标准化方法的特点。 本案例使用的环境为Anaconda Jupyter notebook。 2.数据说明 我们使用的是U…

基于FPGA的超声波测距——UART串口输出

文章目录 前言一、超声波模块介绍1、产品特点2、超声波模块的时序图 二、系统设计1、系统模块框图2、RTL视图 三、源码1、div_clk_us(1us的分频)2、产生驱动超声波的信号3、串口发送模块4、HC_SR04_uart(顶层文件) 四、效果五、总结六、参考资料 前言 环境&#xff1a; 1、Quar…

linux安装python和部署Django项目

文章目录 1 python安装2 Django项目部署 1 python安装 官网地址&#xff1a;https://www.python.org/ 本次下载的python安装包地址&#xff1a;https://www.python.org/ftp/python/3.8.16/Python-3.8.16.tgz 解压下载的python压缩包 [rootlocalhost software]# tar -zxvf P…

Python(五十五)列表元素的修改操作

❤️ 专栏简介&#xff1a;本专栏记录了我个人从零开始学习Python编程的过程。在这个专栏中&#xff0c;我将分享我在学习Python的过程中的学习笔记、学习路线以及各个知识点。 ☀️ 专栏适用人群 &#xff1a;本专栏适用于希望学习Python编程的初学者和有一定编程基础的人。无…

【LeetCode】446. 等差数列划分II -- 子序列

题目链接 文章目录 1. 思路讲解1.1 dp表的创建1.2 状态转移方程1.3 使用哈希表找到k1.4 初始化1.5 返回值1.6 该题坑爹的一点 2. 代码编写 1. 思路讲解 我们要知道以某个位置为结尾的子序列的数量&#xff0c;可以通过它的以上一位置的为结尾的子序列的数量得知&#xff0c;也…

如何微调医疗大模型llm:llama2学习笔记

三个微调方向&#xff1a;简单医疗问答 临床问答 影像学 一般流程&#xff1a; 1 数据集准备 2 模型基座选择 3 微调 4 案例拆解 1 数据集准备&#xff1a;两种类型&#xff0c;一种文本一种影像 扩展&#xff0c;多模态 2 模型基座选择 多模态处理所有视频&#xff0c;文本…