python之链表的反转方式

 更新时间:2023年03月25日 14:18:21   作者:一叶知秋的BLOG  
这篇文章主要介绍了python之链表的反转方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

python链表的反转

反转链表

给你单链表的头节点 head ,请你反转链表,并返回反转后的链表。

  • 输入:head = [1,2,3,4,5]
  • 输出:[5,4,3,2,1]

  • 输入:head = [1,2]
  • 输出:[2,1]

示例 3:

  • 输入:head = []
  • 输出:[]

题解

# Definition for singly-linked list.
# class ListNode:
#     def __init__(self, val=0, next=None):
#         self.val = val
#         self.next = next
class Solution:
    """
    解题思路:
    1.新建一个头指针
    2.遍历head链表,依次在新的头节点位置插入,达到反转的效果
    """
    def reverseList(self, head: ListNode) -> ListNode:
        # 循环
        new_head = None

        while head:
            per = head.next # pre 为后置节点,及当前节点的下一个节点

            head.next = new_head # 插入头节点元素

            new_head = head # 把串起来的链表赋值给头指针

            head = per  # 向后移一个单位
        
        return  new_head  # 返回一个新的链表
                

python反转链表相关技巧

给定一个单链表的头结点pHead(该头节点是有值的,比如在下图,它的val是1),长度为n,反转该链表后,返回新链表的表头。

要求:空间复杂度 O(1)O(1) ,时间复杂度 O(n)O(n) 。

输入:

{1,2,3}

返回值:

{3,2,1}

先来看最基本的反转链表代码:

# -*- coding:utf-8 -*-
# class ListNode:
#     def __init__(self, x):
#         self.val = x
#         self.next = None
class Solution:
    # 返回ListNode
    def ReverseList(self, pHead):
        # write code here
        cur = pHead
        pre = None
        while cur:
            nextNode = cur.next
            cur.next = pre
            pre = cur
            cur = nextNode
        return pre

关键公式

抓住几个关键点:

  • cur:原链表的头节点,在反转结束时,cur指向pre的下一个节点
  • pre:原链表的尾节点,也就是反转后链表的头节点。最终返回的是pre。
  • while cur:表示反转循环的条件,这里是判断cur是否为空。也可以根据题目的条件改成其他循环条件
  • 反转链表的尾节点,这里的尾节点是None,后面会提到显式指定。

对于反转链表的问题,抓住原链表的头节点、原链表的尾节点、反转循环条件、反转链表的尾节点这几个主要角色,基本没什么问题。

接下来,举两个例子:

链表内指定区间反转

链表中的节点每k个一组翻转

链表内指定区间反转

将一个节点数为 size 链表 m 位置到 n 位置之间的区间反转,要求时间复杂度 O(n),空间复杂度 O(1)。

要求:时间复杂度 O(n) ,空间复杂度 O(n)

进阶:时间复杂度 O(n),空间复杂度 O(1)

输入:

{1,2,3,4,5},2,4

返回值:

{1,4,3,2,5}

套用公式

这道题目和baseline的区别是,是将对整个链表的反转改成链表 m 位置到 n 位置之间的区间反转,来套一下公式:

  • 原链表的头节点:cur:从head出发,再走m-1步,到达cur
  • 原链表的尾节点:pre:cur前面的节点
  • 反转循环条件:for i in range(n,m)
  • 反转链表的尾节点:需要保存下从head出发,再走m-1步,到达cur时,此时pre的位置 prePos。prePos.next是反转链表的尾节点

和前面的比,需要额外注意下:

  • 需要保存下从head出发,再走m-1步,到达cur时,此时pre的位置 prePos。在反转循环结束后,再进行穿针引线
  • 由于不是对整个链表进行反转,最好新建虚拟头节点dummpyNode,dummpyNode.next指向整个链表

代码实现

先看下套公式部分的代码:

# 找到pre和cur
i = 1
while i<m:
    pre = cur
    cur = cur.next
    i = i+1
 
# 在指定区间内反转
preHead = pre
while i<=n:
    nextNode = cur.next
    cur.next = pre
    pre = cur
    cur = nextNode
    i = i+1
 

穿针引线部分代码:

nextNode = preHead.next
preHead.next = pre
if nextNode:
    nextNode.next = cur
 

完整代码:

class ListNode:
    def __init__(self, x):
        self.val = x
        self.next = None
 
class Solution:
    def reverseBetween(self , head , m , n ):
        # write code here
        dummpyNode = ListNode(-1)
        dummpyNode.next = head
        pre = dummpyNode
        cur = head
 
        i = 1
        while i<m:
            pre = cur
            cur = cur.next
            i = i+1
 
        preHead = pre
        while i<=n:
            nextNode = cur.next
            cur.next = pre
            pre = cur
            cur = nextNode
            i = i+1
        
        nextNode = preHead.next
        preHead.next = pre
        if nextNode:
            nextNode.next = cur
 
        return dummpyNode.next

链表中的节点每k个一组翻转

将给出的链表中的节点每 k 个一组翻转,返回翻转后的链表

如果链表中的节点数不是 k 的倍数,将最后剩下的节点保持原样

你不能更改节点中的值,只能更改节点本身。

要求空间复杂度 O(1),时间复杂度 O(n)

输入:

{1,2,3,4,5},2

返回值:

{2,1,4,3,5}

套用公式

这道题目和baseline的区别是,是将对整个链表的反转改成每k个一组反转,如果节点数不是k的倍数,剩下的节点保持原样。

先分段来看,假设面对位置1-位置k的链表:

  • 原链表的头节点:cur:从head出发,再走k-1步,到达cur
  • 原链表的尾节点:pre:cur前面的节点
  • 反转循环条件:for i in range(1,k)
  • 反转链表的尾节点:先定义tail=head,等反转完后tail.next就是反转链表的尾节点

先看下套公式部分的代码:

pre = None
cur = head
tail = head
 
 
i = 1
while i<=k:
    nextNode = cur.next
    cur.next = pre
    pre = cur
    cur = nextNode
    i = i+1

这样,我们就得到了1 位置1-位置k的反转链表。

此时:

  • pre:指向反转链表的头节点
  • cur:位置k+1的节点,下一段链表的头节点
  • tail:反转链表的尾节点

那么,得到位置k+1-位置2k的反转链表,就可以用递归的思路,用tail.next=reverse(cur,k)

需要注意:如果链表中的节点数不是 k 的倍数,将最后剩下的节点保持原样

i = 1
tmp = cur
while i<=k:
    if tmp:
        tmp = tmp.next
    else:
        return head
    i = i+1

代码实现

完整代码:

class ListNode:
    def __init__(self, x):
        self.val = x
        self.next = None
 
class Solution:
    def reverseKGroup(self , head , k ):
       
        # write code here
        return self.reverse(head, k )
    
    def reverse(self , head , k ):
        pre = None
        cur = head
        tail = head
 
        i = 1
        tmp = cur
        while i<=k:
            if tmp:
                tmp = tmp.next
            else:
                return head
            i = i+1
        
        i = 1
        while i<=k:
            nextNode = cur.next
            cur.next = pre
            pre = cur
            cur = nextNode
            i = i+1
 
        tail.next = self.reverse(cur, k)
        return pre

好了,抓住几个关键点:

  • cur:原链表的头节点,在反转结束时,cur指向pre的下一个节点
  • pre:原链表的尾节点,也就是反转后链表的头节点。最终返回的是pre。
  • while cur:表示反转循环的条件,这里是判断cur是否为空。也可以根据题目的条件改成其他循环条件
  • 反转链表的尾节点,这里的尾节点是None,后面会提到显式指定。

想清楚这几个关键点都是如何定义的,基本题目都可以迎刃而解啦。

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • 关于python变量的引用以及在底层存储原理

    关于python变量的引用以及在底层存储原理

    Python的变量,简单来说有数值型,布尔型,字符串类型,列表,元组,字典等6大类。那么不同变量类型在底层是如何存储的,关系到变量的引用,能否正确的掌握变量的相关操作?接下来小编就来为大家讲解python变量的引用以及在底层存储原理,需要的朋友可以参考一下
    2021-09-09
  • 基于Python实现图像文字识别OCR工具

    基于Python实现图像文字识别OCR工具

    在工作、生活中常常会用到,比如票据、漫画、扫描件、照片的文本提取。本文主要介绍了基于PyQt + PaddleOCR实现的一个桌面端的OCR工具,用于快速实现图片中文本区域自动检测+文本自动识别,需要的朋友可以参考一下
    2021-12-12
  • 一篇文章搞懂python混乱的切换操作与优雅的推导式

    一篇文章搞懂python混乱的切换操作与优雅的推导式

    这篇文章主要给大家介绍了如何通过一篇文章搞懂python混乱的切换操作与优雅的推导式的相关资料,文中通过示例代码介绍的非常详细,对大家的学习具有一定的参考学习价值,需要的朋友可以参考下
    2021-08-08
  • Python3.9环境搭建RobotFramework的详细过程

    Python3.9环境搭建RobotFramework的详细过程

    Robot Framework是一个基于Python的,可扩展的关键字驱动的测试自动化框架,用于端到端验收测试和验收测试驱动开发(ATDD),这篇文章主要介绍了Python3.9环境搭建RobotFramework的详细过程,需要的朋友可以参考下
    2023-01-01
  • LyScript实现Hook隐藏调试器的方法详解

    LyScript实现Hook隐藏调试器的方法详解

    LyScript 插件集成的内置API函数可灵活的实现绕过各类反调试保护机制。本文将运用LyScript实现绕过大多数通用调试机制,实现隐藏调试器的目的,需要的可以参考一下
    2022-09-09
  • python 实现上传图片并预览的3种方法(推荐)

    python 实现上传图片并预览的3种方法(推荐)

    下面小编就为大家带来一篇python 实现上传图片并预览的3种方法(推荐)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-07-07
  • pandas将多个dataframe以多个sheet的形式保存到一个excel文件中

    pandas将多个dataframe以多个sheet的形式保存到一个excel文件中

    这篇文章主要介绍了pandas将多个dataframe以多个sheet的形式保存到一个excel文件中,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-10-10
  • 使用Python生成随机图片验证码的代码详解

    使用Python生成随机图片验证码的代码详解

    当我们在写一个Web项目的时候一般要写登录操作,而为了安全起见,现在的登录功能都会加上输入图片验证码这一功能,所以本文就给大家介绍一下如何使用Python生成随机图片验证码,需要的朋友可以参考下
    2023-07-07
  • Python 连接 MySQL 的几种方法

    Python 连接 MySQL 的几种方法

    这篇文章主要介绍了Python 连接 MySQL 的几种方法,帮助大家更好的理解和使用python,感兴趣的朋友可以了解下
    2020-09-09
  • python用字节处理文件实例讲解

    python用字节处理文件实例讲解

    在本篇文章里小编给大家整理的是一篇关于python用字节处理文件实例讲解内容,有兴趣的朋友们可以学习参考下。
    2021-04-04

最新评论