DAY04算法学习|链表part02 24. 两两交换链表中的节点 , 19.删除链表的倒数第N个节点 , 面试题 02.07. 链表相交 , 142.环形链表II

本文介绍了三个基于链表的编程问题解决方案,包括两两交换链表中的节点、删除链表的倒数第N个节点以及找出两个链表相交的起始节点。解题思路涉及创建虚拟头节点、遍历链表和双指针技术,同时分析了每个问题的时间和空间复杂度。
摘要由CSDN通过智能技术生成

题目 24. 两两交换链表中的节点

题目链接

题目描述:

给定一个链表,交换每两个相邻节点并返回它的头。 您必须在不修改列表节点中的值的情况下解决问题(即,只能更改节点本身。)

解题思路:

  1. 为了处理方便,我们先初始化一个预头节点 res,并且设置一个当前节点 pre 指向预头节点。

  2. 然后我们进入循环,判断 pre.nextpre.next.next 是否存在。只有这两个节点都存在,我们才能进行交换。

  3. 交换的过程主要包括以下步骤:

    • 首先我们需要记录下 pre.next(我们称之为节点1)和 pre.next.next(我们称之为节点2)。
    • 然后我们需要将节点1链接到节点2的下一个节点,这一步是为了防止链表断裂。
    • 接着我们需要将当前节点 pre 链接到节点2,完成节点2与前面节点的连接。
    • 最后我们需要将节点2链接到节点1,完成两个节点的交换。
  4. 交换完成后,我们需要移动当前节点 pre 到下一组需要交换的节点的前面,也就是 pre.next.next

  5. 最后我们返回预头节点的下一个节点,也就是交换后链表的头节点。

代码:

创造一个虚拟头节点,写出限制条件

pre.next!=None and pre.next.next!=None

# Definition for singly-linked list.
# class ListNode:
#     def __init__(self, val=0, next=None):
#         self.val = val
#         self.next = next
class Solution:
    def swapPairs(self, head: Optional[ListNode]) -> Optional[ListNode]:
        res=ListNode(next=head)
        pre=res
        # because first reverse point must be next to the cur
        while(pre.next!=None and pre.next.next!=None):
            temp1=pre.next
            temp2=pre.next.next
            #node1 point to node3
            temp1.next=temp2.next
           
            #cur point to node2
            pre.next=temp2
            #node2 point to node1
            temp2.next=temp1
            #move cur to a new round
            pre=pre.next.next
        return res.next
        



            
            


        


        

复杂度:

时间复杂度:O(n),我们需要遍历所有的节点,进行交换,所以时间复杂度是 O(n),n 是链表的长度。

空间复杂度:O(1),我们只使用了常数级别的额外空间,所以空间复杂度是 O(1)。

题目 19.删除链表的倒数第N个节点

题目链接

题目描述:

给定一个链表的头,从链表的末尾移除第 n 个节点并返回它的头。

解题思路:

  1. 首先,我们需要获取链表的长度,可以通过一次遍历完成。

  2. 其次,我们初始化一个预头节点 head_dummy,并将其 next 指向 head。这是为了处理特殊情况,例如当需要删除的节点就是头节点时。

  3. 然后,我们计算出需要移动的次数 move_time,这个次数等于链表的长度 len 减去 n。也就是说,我们需要移动到倒数第 n 个节点的前一个节点。

  4. 接着,我们移动 cur 到指定的节点,然后修改它的 next 指针,使其指向下下个节点(也就是跳过下一个节点,达到删除的效果)。

  5. 最后,我们返回 head_dummy.next,也就是处理后的链表的头节点。

代码:

# Definition for singly-linked list.
# class ListNode:
#     def __init__(self, val=0, next=None):
#         self.val = val
#         self.next = next
class Solution:
    def removeNthFromEnd(self, head: Optional[ListNode], n: int) -> Optional[ListNode]:
        #get the length of linkedList
        current=head
        len=0
        while(current!=None):
            current=current.next
            len+=1
        head_dummy=ListNode()
        head_dummy.next=head
        cur=head_dummy
        #reach the node before the target node, slow/fast nend to move 3(len-n) time
        move_time=len-n
        while(move_time):
            cur=cur.next
            move_time-=1
        cur.next=cur.next.next
        return head_dummy.next


            

复杂度:

题目 160链表相交

题目链接

题目描述:

给你两个单链表的头节点 headA 和 headB ,请你找出并返回两个单链表相交的起始节点。如果两个链表没有交点,返回 null 。

图示两个链表在节点 c1 开始相交:

题目数据 保证 整个链式结构中不存在环。

注意,函数返回结果后,链表必须 保持其原始结构 。

示例 1:

输入:intersectVal = 8, listA = [4,1,8,4,5], listB = [5,0,1,8,4,5], skipA = 2, skipB = 3
输出:Intersected at '8'
解释:相交节点的值为 8 (注意,如果两个链表相交则不能为 0)。
从各自的表头开始算起,链表 A 为 [4,1,8,4,5],链表 B 为 [5,0,1,8,4,5]。
在 A 中,相交节点前有 2 个节点;在 B 中,相交节点前有 3 个节点。

解题思路:

  1. 首先,我们需要计算链表A和链表B的长度。这可以通过定义一个获取链表长度的函数 getlen 来实现。

  2. 然后,我们创建两个预头节点 dummy_nodeAdummy_nodeB,并将它们分别指向链表A和链表B的头节点。

  3. 接下来,我们比较两个链表的长度。假设链表A的长度大于或等于链表B,我们就需要移动链表A的预头节点 dummy_nodeA,使其移动的步数等于两个链表长度的差值。如果链表B的长度大于链表A,我们就需要移动链表B的预头节点 dummy_nodeB

  4. 接着,我们同时移动两个链表的预头节点,直到它们指向的节点相同,也就是找到了相交的起始节点。

  5. 最后,我们返回相交的起始节点,如果没有找到相交节点,就返回 None。

代码:

# Definition for singly-linked list.
# class ListNode:
#     def __init__(self, x):
#         self.val = x
#         self.next = None

class Solution:
    def getIntersectionNode(self, headA: ListNode, headB: ListNode) -> Optional[ListNode]:
        #write a len function
        def getlen(node):
            cur=node
            len_=0
            while node:
                len_+=1
                node=node.next
            return len_
        lenA=getlen(headA)
        lenB=getlen(headB)

        #create dummy_node for both A and B
        dummy_nodeA=ListNode(next=headA)
        dummy_nodeB=ListNode(next=headB)

        #compare the length to know move which first
        if lenA>=lenB:
            stepA=lenA-lenB
            while(stepA):
                dummy_nodeA=dummy_nodeA.next
                stepA-=1
            while(dummy_nodeA.next!=dummy_nodeB.next):
                dummy_nodeA=dummy_nodeA.next
                dummy_nodeB=dummy_nodeB.next
            if dummy_nodeA.next==dummy_nodeB.next:
                return dummy_nodeA.next
            else:
                return None

        else:
            stepB=lenB-lenA
            while (stepB):
                dummy_nodeB=dummy_nodeB.next
                stepB-=1
            while(dummy_nodeA.next!=dummy_nodeB.next):
                dummy_nodeA= dummy_nodeA.next
                dummy_nodeB=dummy_nodeB.next
            if dummy_nodeB.next== dummy_nodeA.next:
                return dummy_nodeB.next
            else:
                return None
        
        


            
        

复杂度:

时间复杂度:O(n+m),我们需要遍历两个链表计算长度,再遍历链表找到相交的起始节点,所以时间复杂度是 O(n+m),n 和 m 分别是两个链表的长度。

空间复杂度:O(1),我们只使用了常数级别的额外空间,所以空间复杂度是 O(1)。

题目 142环形链表II 

题目链接

题目描述:

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

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

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

解题思路:

代码随想录解析

  1. 首先,我们定义两个指针,一个是快指针 fast,一个是慢指针 slow。快指针每次走两步,慢指针每次走一步。

  2. 当快指针和慢指针相遇时,说明链表中存在环。此时,我们定义两个新的指针 index1index2index1 从链表头开始,index2 从相遇点开始,两个指针都每次走一步。

  3. index1index2 再次相遇时,相遇的节点就是环的起始节点。

代码:

# Definition for singly-linked list.
# class ListNode:
#     def __init__(self, x):
#         self.val = x
#         self.next = None

class Solution:
    def detectCycle(self, head: Optional[ListNode]) -> Optional[ListNode]:
        #define fast and slow
        slow,fast=head,head
        while fast and fast.next:
            slow=slow.next
            fast=fast.next.next
            if slow==fast:
                index1=head
                index2=slow
                while(index1!=index2):
                    index1=index1.next
                    index2=index2.next
                return index1

复杂度:

时间复杂度:O(n),我们最多遍历链表两次,所以时间复杂度是 O(n),n 是链表的长度。

空间复杂度:O(1),我们只使用了常数级别的额外空间,所以空间复杂度是 O(1)。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值