147. 对链表进行插入排序

147. 对链表进行插入排序

原始题目链接:https://leetcode-cn.com/problems/insertion-sort-list/

在这里插入图片描述

解题思路:

类似于数组或者列表的插入排序,但题目操作的是单链表,单链表的插入排序,要知道这几个信息:操作的元素、插入的位置、插入位置的前驱节点;其次需要注意的是插入节点的时候,操作多个节点的指针指向的顺序,在代码的注释中有说明,做链表的题目最好是画图,画图就很好解题了,具体看代码及注释。

在这里插入图片描述

代码实现:

# Definition for singly-linked list.
# class ListNode:
#     def __init__(self, val=0, next=None):
#         self.val = val
#         self.next = next
class Solution:
    def insertionSortList(self, head: ListNode) -> ListNode:
        if head is None:
            return None

        # 设置一个哑节点,方便返回修改后的链表和原始链表的头结点也可能被修改
        # 数据域初始化为-1,指针域指向原始链表的头结点
        dummy = ListNode(-1)
        dummy.next = head

        # 需要定义几个指针用于插入的操作
        # 排好序的链表中最后一个节点
        last_sorted_node = head

        # 要操作的节点,就是需要重新插入适当位置的节点
        unsorted_node = head.next

        # 直到没有排序的节点都遍历完,则结束循环
        while unsorted_node:
            # 正常顺序,节点前移
            if last_sorted_node.val <= unsorted_node.val:
                last_sorted_node = last_sorted_node.next
            else:
                # unsorted_node节点的前驱节点,因为单链表插入节点的操作需要知道插入位置的前驱点
                # 初始化为哑节点,是因为原始链表的头结点也可能会改变,初始为head是不正确的
                # 因为单链表就是需要从头每次查找要插入的位置,所以前驱节点每次都初始化为dummy
                pre_node = dummy
                
                # 查找插入位置的前驱节点
                while pre_node.next.val <= unsorted_node.val:
                    pre_node = pre_node.next
                # 开始改变指针的指向,注意这里的顺序
                last_sorted_node.next = unsorted_node.next
                unsorted_node.next = pre_node.next
                pre_node.next = unsorted_node
            
            # 每次都调整未排序节点的指向,保证指向的是排好序节点的下一个节点
            # 这样才能比较并找到要插入的是哪个节点
            unsorted_node = last_sorted_node.next

        return dummy.next

参考文献:
https://leetcode-cn.com/problems/insertion-sort-list/solution/dui-lian-biao-jin-xing-cha-ru-pai-xu-by-leetcode-s/

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值