前言

给定一个单链表的头结点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:# 返回ListNodedef ReverseList(self, pHead):# write code herecur = pHeadpre = Nonewhile cur:nextNode = cur.nextcur.next = prepre = curcur = nextNodereturn 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 = curcur = cur.nexti = i+1# 在指定区间内反转
preHead = pre
while i<=n:nextNode = cur.nextcur.next = prepre = curcur = nextNodei = i+1

穿针引线部分代码:

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

完整代码:

class ListNode:def __init__(self, x):self.val = xself.next = Noneclass Solution:def reverseBetween(self , head , m , n ):# write code heredummpyNode = ListNode(-1)dummpyNode.next = headpre = dummpyNodecur = headi = 1while i<m:pre = curcur = cur.nexti = i+1preHead = prewhile i<=n:nextNode = cur.nextcur.next = prepre = curcur = nextNodei = i+1nextNode = preHead.nextpreHead.next = preif nextNode:nextNode.next = curreturn 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 = headi = 1
while i<=k:nextNode = cur.nextcur.next = prepre = curcur = nextNodei = 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.nextelse:return headi = i+1

代码实现

完整代码:

class ListNode:def __init__(self, x):self.val = xself.next = Noneclass Solution:def reverseKGroup(self , head , k ):# write code herereturn self.reverse(head, k )def reverse(self , head , k ):pre = Nonecur = headtail = headi = 1tmp = curwhile i<=k:if tmp:tmp = tmp.nextelse:return headi = i+1i = 1while i<=k:nextNode = cur.nextcur.next = prepre = curcur = nextNodei = i+1tail.next = self.reverse(cur, k)return pre

总结

抓住几个关键点:

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

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

[Python] 反转链表相关技巧相关推荐

  1. python 反转链表

    反转链表 定义一个函数,输入一个链表的头节点,反转该链表并输出反转后链表的头节点.示例:输入: 1->2->3->4->5->NULL 输出: 5->4->3 ...

  2. python反转链表_206. 反转链表(Python)

    题目 难度:★★☆☆☆ 类型:链表 反转一个单链表. 进阶: 你可以迭代或递归地反转链表.你能否用两种方法解决这道题? 示例 输入: 1->2->3->4->5->NUL ...

  3. python反转链表和成对反转

    https://www.cnblogs.com/tianqizhi/p/9673894.html https://blog.csdn.net/weixin_34168700/article/detai ...

  4. python实现链表的删除_干货||链表的技巧和算法总结

    链表的操作总结   链表反转 这是一个简单的链表操作问题,在leetcode上面有52.7%的通过率,难度是简单.但是还是想在这里基于python做一下总结,顺便总结一下链表的各种操作. 首先先看一下 ...

  5. python实现反转链表讲解_基于Python实现2种反转链表方法代码实例

    题目: 反转一个单链表. 示例: 输入: 1->2->3->4->5->NULL 输出: 5->4->3->2->1->NULL 进阶: 你 ...

  6. python实现链表反转(转置)

    python实现链表反转(转置) 链表是面试里面经常涉及到的考点,因为链表的结构相比于Hashmap.Hashtable.Concurrenthashmap或者图等数据结构简单许多,对于后者更多面试的 ...

  7. 【剑指Offer专题】链表系列:从尾到头打印链表、反转链表、回文链表、合并两个排序的链表(C++和Python实现)...

    关注上方"深度学习技术前沿",选择"星标公众号", 资源干货,第一时间送达! 剑指Offer(三):从尾到头打印链表 输入一个链表的头节点,从尾到头反过来返回每 ...

  8. python链表翻转_反转链表(两种Python解法)

    题目: 反转一个单链表. 示例: 输入: 1->2->3->4->5->NULL 输出: 5->4->3->2->1->NULL 进阶: 你 ...

  9. 翻转链表python递归_Python LeetCode-206.反转链表(难度-简单) 两个方法-迭代和递归,以及超简写法(python)...

    1.题目描述 反转一个单链表. 实例 输入: 1->2->3->4->5->NULL 输出: 5->4->3->2->1->NULL 进阶: ...

最新文章

  1. 解决mac下webstorm编辑器识别less的问题
  2. ReentrantLock可以是公平锁,sync只能是非公平锁。
  3. wxWidgets:wxStringBufferLength类用法
  4. P4111 [HEOI2015]小Z的房间
  5. 自动加密企业关键业务数据 赛门铁克推出全新信息保护解决方案
  6. linux隐藏apache信息,Linux下如何隐藏Apache版本号信息
  7. Detectron2 win10踩坑记录
  8. java显示行号_java开发文本编辑器手记之行号显示
  9. wordpress修改mysql端口_Wordpress更改端口后访问自动跳转老端口
  10. 安装Bas的先决条件
  11. ssh client 报 algorithm negotiation failed的解决方法
  12. Zepto:基础学习
  13. 织梦本地调试运行PHP不显示图片,织梦CMS手机端不显示图片的原因及解决方法!...
  14. 定时 监控 shell 服务宕机自动重启,并发送短信通知
  15. javascript中的字符串编码转换
  16. 小游戏《恶梦》的总结
  17. Unity3D--控制鼠标的显示与隐藏
  18. tomcat开启远程调试功能
  19. NOI2016酱油记
  20. 关于在打败C魔王前经常被背刺这件事(C语言常见问题描述及解决方案和原因)

热门文章

  1. Ubuntu好用的词典 星际译王 stardict安装 词典添加
  2. 电脑cpu风扇转一下就停无法开机_win7系统电脑开机cpu风扇转一下就停下来怎么办...
  3. 「数据架构」5分钟学会数据流程图:客户服务系统示例
  4. 【pythonCAN】基于Kvaser开发CAN总线分析工具
  5. win7系统安装office 2010 visio 出现MSXML安装后依旧无法安装的解决办法
  6. playsound 模块解决 UnicodeDecodeError 异常
  7. RGB565转RGB888
  8. IOS xib三等分,四等分控件
  9. 内网搭建WEB服务器教程(转载)
  10. 工程监测多通道振弦模拟信号采集仪VTN数据查看