python之链表的反转方式
作者:一叶知秋的BLOG 发布时间:2022-02-08 10:20:06
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,后面会提到显式指定。
想清楚这几个关键点都是如何定义的,基本题目都可以迎刃而解啦。
来源:https://blog.csdn.net/python_tian/article/details/121342104


猜你喜欢
- 一、简介Selenium是一个用于Web应用程序测试的工具。Selenium测试直接运行在浏览器中,就像真正的用户在操作一样。支持的浏览器包
- 前言本文中所有的代码使用 JavaScript 编写,但你也可以用其他兼容 JSR 223 的脚本语言。这些例子可作为脚本文件也可以在交互式
- 这两天终于忍不住的去实验了一下,为什么网页的字体有时会显示成超级无敌难看的宋体呢?其实宋体不难看,难看的只是把它放在Leopard下,没有点
- 将程序转换为exe文件我们先来介绍如何使用工具Pyinstaller安装Pyinstaller我们用pip安装Pyinstaller 。注意
- 前段时间开始学习JSON在ASP中的使用,JSON确实比XML方便。以前在写程序的时候有考虑使用XML来存储数据,但是一直觉得生成文档及文档
- 前情提要好久没有写Vue了,略有生疏,这个东西还是得多用。下午看到一个需求,选择相册图片作为轮播图显示。接口返回相册列表,用户选一下再扔回去
- 本文讲述了Javascript简单实现面向对象编程继承实例代码。分享给大家供大家参考,具体如下:面向对象的语言必须具备四个基本特征:1.封装
- 在开发软件的过程中,我们经常会碰到需要在指定目录下生成文件和删除文件的操作,下面就演示一下怎样用python进行之类操作。生成文件impor
- 今天在写爬虫程序的时候由于要翻页,做除法分页的时候出现了totalCount = '100' totalPage = int
- 本文实例讲述了python实现在字符串中查找子字符串的方法。分享给大家供大家参考。具体如下:这里实现python在字符串中查找子字符串,如果
- 出现问题: 1. 使用层制作的下拉菜单下正好有FLASH动画,菜单被动画遮挡. 2. 页面中的层浮动广告当经过FLASH动画时,浮动层从动画
- 矩阵增加行np.row_stack() 与 np.column_stack()import numpy as npa = np.array(
- mysql删除操作其实是假删除在 InnoDB 中,你的 delete 操作,并不会真的把数据删除,mysql 实际上只是给删除的数据打了个
- 可以通过遍历的方法:pandas按行按列遍历Dataframe的几种方式:https://www.jb51.net/article/1726
- MySQL是一种关系型数据库管理系统,关系数据库将数据保存在不同的表中,而不是将所有数据放在一个大仓库内,这样就增加了速度并提高了灵活性。M
- 注:MySQL5.7破解root密码,跳过密码认证登录到数据库,直接修改表中的密码即可,但是MySQL 8.0则不可以这样修改root密码,
- 图例如下https://github.com/Dongvdong/python_Smartvoice上电后,只要周围声音超过 2000,开始
- 说到排序,很多人可能第一想到的就是sorted,但是你可能不知道python中其实还有还就中方法哟,并且好多种场景下效率都会比sorted高
- javascript是种脚本语言,浏览器下载到哪儿就会执行到哪儿,这种特性会为编程提供方便,但也容易使程序过于凌乱,支离破碎。 js从功能上
- 将datetime64[ns]转为字符串日期将datetime64[ns]转为字符串日期(“%Y-%m-%d&r