Smaller And Smarter Python数据结构:链表进行重新排序

简介: Smaller And Smarter Python数据结构:链表进行重新排序

简说Python,号主老表,Python终身学习者,数据分析爱好者,从18年开始分享Python知识,原创文章227篇,写过Python、SQL、Excel入门文章,也写过Web开发、数据分析文章,老表还总结整理了一份2022Python学习资料和电子书资源,关注后私信回复:2022 即可领取。

今天给大家分享的书籍《Python程序员面试算法宝典》第一章第四小节:链表进行重新排序。

如果你是第一次看,也许,你可以看看本系列下面的文章:

Smaller And Smarter Python数据结构:链表逆转

Smaller And Smarter Python数据结构:删除无序链表重复结点

Smaller And Smarter Python数据结构:链表相加

今日问题

"""
目标:写一段程序,对链表进行重新排序
例如:
输入-> L1->L2->L3->...->Ln-1->Ln
输出-> L1->Ln->L2->Ln-1...
"""
"""
Objective: write a program to reorder the linked list
For example:
Input - > L1 - > L2 - > L3 -> ... - > ln-1 - > LN
Output - > L1 - > ln - > L2 - > ln-1...
"""

题目要求

(1)在原来链表的基础上进行排序,即不能申请新的结点;

(2)只能修改结点的next域,不能修改数据域。

首先我们写好链表的基本操作,在a_0_base.py文件中,目前包含对链表的定义类,初始化函数,遍历函数。(带头结点)

# -*- coding: utf-8 -*-
"""
@author = 老表
@date = 2019-10-19
@个人微信公众号 : 简说Python
"""
# 链表数据结构定义
class ListNode:
    def __init__(self, x):
        self.data = x
        self.next = None
class ListOperation:
    # 根据链表数据初始化链表
    @staticmethod
    def init_list(n_list):
        # 初始化一个头指针
        head = ListNode("head")
        cur = head
        for i in n_list:
            node = ListNode(i)
            cur.next = node
            cur = node  # 相当于 cur = cur.next,后移
        return head
    # 遍历链表,带头结点
    @staticmethod
    def ergodic_list(head):
        cur = head.next
        while cur:
            print(cur.data)
            cur = cur.next

解题

开始程序前需提前导入前面写好的链表基本操作包,在Linked_list的a_0_base.py中。

from Linked_list.a_0_base import ListOperation

思路分析

"""
核心思想:先将链表前 后段分开,然后将后半段链表逆转,
再遍历链表,将处理好的后半段链表结点插入到前半段链表中。
优点:该题比较综合,遍历链表,逆转链表,插入结点
时间复杂度:O(N)
空间复杂度:O(1)
"""

注意:代码中的注释部分可以不需要,是我自己编写、调试bug时用的,之所以留在上面,主要为了方便大家调试代码和理解代码。

辅助函数:查找链表中间结点

# 辅助函数:找出链表中间结点
def find_mid_link(head):
    fast_node = head.next  # 快指针,每次后移两位
    slow_node = head.next  # 慢指针,每次后移一位
    while fast_node and fast_node.next:  # 遍历链表,快指针遍历结束时,慢指针应该刚好遍历到一半
        pre_slow_node = slow_node  # 记录慢指针前驱结点,方便断开前后半部分链表
        slow_node = slow_node.next  # 慢指针后移,步进1
        fast_node = fast_node.next.next  # 快指针后移,步进2
    pre_slow_node.next = None  # 断开前半部分链表与后半部分链表连接
    return slow_node  # 返回中间结点

辅助函数:不带头结点链表逆转

# 辅助函数:不带头结点链表逆转
def reverse_list(l1):
    if not l1 or not l1.next:  # 空链表或只有一个结点
        return l1
    cur_node = l1.next  # 记录当前结点
    pre_node = l1  # 记录前驱结点
    l1.next = None  # 设置尾结点
    # print("------------------")
    while cur_node:
        # print(cur_node.data)
        next_node = cur_node.next  # 记录后继结点
        cur_node.next = pre_node  # 当前结点反向指向前驱结点
        pre_node = cur_node  # 前驱结点后移
        cur_node = next_node  #  当前结点后移
        # print(pre_node.data)
    return pre_node  # 返回链表表头结点

主功能函数

def reorder_link_one(head):
    """
    :type head: ListNode
    :rtype: ListNode
    """
    if not head.next:  # 空链表
        return head  # 返回头结点
    mid_node = find_mid_link(head)  # 查找链表中间结点
    # print("------------------")
    # s0 = ListOperation()
    # s0.ergodic_list(mid_node)
    # print("------------------")
    # s0.ergodic_list(head)
    mid_node = reverse_list(mid_node)  # 逆转后半部分链表,无头结点
    # print("------------------")
    # ListOperation.ergodic_list(mid_node)
    cur_node = head.next  # 记录前半段链表第一个结点
    while cur_node.next:  # 遍历前半段结点,将后半段结点插入
        temp_node = cur_node.next  # 记录后继结点
        cur_node.next = mid_node  # 插入
        cur_node = temp_node  # 后移
        temp_node = mid_node.next  # 记录后继结点
        mid_node.next = cur_node  # 插入
        mid_node = temp_node  # 后移
    cur_node.next = mid_node  # 将mid_node最后一个结点插入
    return head  # 返回当前子链表头结点

测试代码

# 当然,也许还有别的方法,比如建一个辅助的链表
# 欢迎你说出你的想法
# 程序入口,测试函数功能
if __name__ == "__main__":
    s0 = ListOperation()  # 初始化一个链表基本操作对象实例
    list_data = [1, 2, 3, 4, 5, 6]  # 链表初始化数据
    head = s0.init_list(list_data)  # 初始化链表,带头结点
    s0.ergodic_list(head)  # 未操作前,遍历打印链表
    head = reorder_link_one(head)  # 测试方法一
    print("---------------------")
    s0.ergodic_list(head)  # 操作后,遍历打印链表

本文代码思路部分来自书籍《Python程序员面试宝典》,书中部分代码有问题或未提供代码,文中已经修改过了,并添加上了丰厚的注释,方便大家学习,后面我会把所有内容开源放到Github上,包括代码,思路,算法图解(手绘或者电脑画),时间充裕的话,会录制视频。

希望大家多多支持。

大家好,我是老表

觉得本文不错的话,转发、留言、点赞,是对我最大的支持。

相关文章
|
24天前
|
存储 Python
Python 实现单向链表,和单向链表的反转
链表是一种数据结构,每个节点存储相邻节点的位置信息。单链表中的节点仅存储下一节点的位置。通过Python实现单链表,定义`ListNode`类并关联节点可创建链表。例如,创建A->B->C的链表后,可通过反转函数`reverse`将链表反转为CBA。代码展示了如何实现和操作单链表。
Python 实现单向链表,和单向链表的反转
|
1月前
|
存储 机器学习/深度学习 算法
C 408—《数据结构》算法题基础篇—链表(下)
408考研——《数据结构》算法题基础篇之链表(下)。
96 29
|
3月前
|
存储 缓存 监控
局域网屏幕监控系统中的Python数据结构与算法实现
局域网屏幕监控系统用于实时捕获和监控局域网内多台设备的屏幕内容。本文介绍了一种基于Python双端队列(Deque)实现的滑动窗口数据缓存机制,以处理连续的屏幕帧数据流。通过固定长度的窗口,高效增删数据,确保低延迟显示和存储。该算法适用于数据压缩、异常检测等场景,保证系统在高负载下稳定运行。 本文转载自:https://www.vipshare.com
140 66
|
1月前
|
存储 算法 C语言
C 408—《数据结构》算法题基础篇—链表(上)
408考研——《数据结构》算法题基础篇之链表(上)。
121 25
|
2月前
|
数据挖掘 数据处理 开发者
Python3 自定义排序详解:方法与示例
Python的排序功能强大且灵活,主要通过`sorted()`函数和列表的`sort()`方法实现。两者均支持`key`参数自定义排序规则。本文详细介绍了基础排序、按字符串长度或元组元素排序、降序排序、多条件排序及使用`lambda`表达式和`functools.cmp_to_key`进行复杂排序。通过示例展示了如何对简单数据类型、字典、类对象及复杂数据结构(如列车信息)进行排序。掌握这些技巧可以显著提升数据处理能力,为编程提供更强大的支持。
57 10
|
1月前
|
存储 算法 搜索推荐
Python 实现反转、合并链表有啥用?
大家好,我是V哥。本文介绍Python实现反转链表和合并链表的应用场景及代码实现。反转链表适用于时间序列数据展示、回文链表判断等;合并链表则用于大规模数据排序、数据库查询结果集合并等。通过迭代和递归方法实现反转链表,以及合并两个或多个有序链表的算法,帮助开发者解决实际问题。关注V哥,了解更多实用编程技巧。 先赞再看后评论,腰缠万贯财进门。
|
2月前
|
机器学习/深度学习 存储 C++
【C++数据结构——线性表】单链表的基本运算(头歌实践教学平台习题)【合集】
本内容介绍了单链表的基本运算任务,涵盖线性表的基本概念、初始化、销毁、判定是否为空表、求长度、输出、求元素值、按元素值查找、插入和删除数据元素等操作。通过C++代码示例详细解释了顺序表和链表的实现方法,并提供了测试说明、通 - **任务描述**:实现单链表的基本运算。 - **相关知识**:包括线性表的概念、初始化、销毁、判断空表、求长度、输出、求元素值、查找、插入和删除等操作。 - **测试说明**:平台会对你编写的代码进行测试,提供测试输入和预期输出。 - **通关代码**:给出了完整的C++代码实现。 - **测试结果**:展示了测试通过后的预期输出结果。 开始你的任务吧,祝你成功!
121 5
|
3月前
|
存储 运维 监控
探索局域网电脑监控软件:Python算法与数据结构的巧妙结合
在数字化时代,局域网电脑监控软件成为企业管理和IT运维的重要工具,确保数据安全和网络稳定。本文探讨其背后的关键技术——Python中的算法与数据结构,如字典用于高效存储设备信息,以及数据收集、异常检测和聚合算法提升监控效率。通过Python代码示例,展示了如何实现基本监控功能,帮助读者理解其工作原理并激发技术兴趣。
92 20
|
3月前
|
数据库
数据结构中二叉树,哈希表,顺序表,链表的比较补充
二叉搜索树,哈希表,顺序表,链表的特点的比较
数据结构中二叉树,哈希表,顺序表,链表的比较补充
|
2月前
|
Python
探索 Python 中链表的实现:从基础到高级
链表是一种由节点组成的基础数据结构,每个节点包含数据和指向下一个节点的引用。本文通过Python类实现单向链表,详细介绍了创建、插入、删除节点等操作,并提供示例代码帮助理解。链表在处理动态数据时具有高效性,适用于大量数据变动的场景。文章为初学者提供了全面的入门指南,助你掌握链表的核心概念与应用。
103 0