算法详解——链表的归并排序非递归解法

news2025/1/16 14:05:21

算法详解——链表的归并排序非递归解法

本文使用倍增法加上归并排序操作实现了对链表的快速排序,比起一般的递归式归并排序要节省空间并且实现要简单的多,比起一般的迭代式归并排序实现也要简单。

1. 题目假设

给定链表的头结点 head ,请将其按 升序 排列并返回 排序后的链表

要实现对该链表的快速排序(时间复杂度达到O(nlogn)),比较合适的选择是归并排序(当然快速排序也不是不行)。

image-20241107000431106 ## 2. 回顾一般的解法

一般的解法主要有两种形式,即自顶向下法自底向上法,前者使用递归实现,后者使用迭代实现。

自定向下——递归法

算法思想如下:

  1. 如果当前链表为空或者只有一个结点,直接返回该链表
  2. 找到该链表的中间结点(可以使用快慢指针)
  3. 对左右两条子链执行递归,得到两条排好序的子链
  4. 对两条子链进行归并排序,得到一个有序链表,返回其头指针

自底向上——迭代法

算法思想如下:

  1. 设置步长step为1
  2. 以step个结点为一条子链,从头节点开始遍历,每凑够两条子链,就执行归并操作。如果结点总数小于等于step,直接返回。
  3. 倍增step

但两种方法对于数组来说是比较合适的,因为数组可以随机访问,这样可以很方便的找到中点或者找到下一个长度为step的数组。但是对于链表来说实现起来就比较复杂。

3. 倍增法归并排序

倍增法归并排序的思想是这样的,想象一个64位的整数,那么整数中的第i位能够代表2的i次方的值。

当执行加1操作时,将会从第0位开始如果是0就变为1并返回,如果是1就变为0并继续向后操作。

受此启发,我们可以建立一个64位大小的链表数组,然后遍历待排序链表的每个结点,将其“加入”链表数组。

加入的方法为:从数组的左边开始遍历,如果该位为空就直接加入并返回,如果不为空就执行归并操作并将该位置为空,然后继续向后操作。
在这里插入图片描述

详细操作步骤请看代码和注释。代码如下:

/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode() : val(0), next(nullptr) {}
 *     ListNode(int x) : val(x), next(nullptr) {}
 *     ListNode(int x, ListNode *next) : val(x), next(next) {}
 * };
 */
class Solution {
public:
    ListNode* sortList(ListNode* head) {
        // 创建大小为64的链表指针数组
        ListNode *A[64] = {0};
		
        // 遍历待排序链表的每个结点并将其加入指针数组
        auto it = head;
        while (it != nullptr)
        {
            // 取下当前头节点,并赋值给cur
            auto cur = it;
            it = it->next;
            cur->next = nullptr;
			
    		// 遍历指针数组
            for (int i = 0; i < 64; ++i)
            {
                // 如果当前数组元素为空(对应整数加法中当前位为0),则将其置为当前链表cur
                if (A[i] == nullptr)
                {
                    A[i] = cur;
                    break;
                }
                else	// 否则将cur与当前数组元素所指链表执行归并操作,并将得到的新链表赋给cur,然后将数组元素置空(想想对应加法操作中的什么)
                {
                    cur = merge(cur, A[i]);
                    A[i] = nullptr;
                }
            }
        }
		
        // 最后将指针数组中的所有链表都从左到右进行归并操作,得到一个完整的排好序的链表
        ListNode *res = nullptr;
        for (int i = 0; i < 64; ++i)
        {
            if (A[i] != nullptr)
            {
                res = merge(res, A[i]);
            }
        }
        return res;
    }
	
    // 对left和right两条链表进行归并排序
    ListNode *merge(ListNode *left, ListNode *right)
    {
        ListNode dummy_node;	// 伪头结点
        ListNode *tar = &dummy_node;

        while (left != nullptr & right != nullptr)
        {
            if (left->val < right->val)
            {
                tar->next = left;
                left = left->next;
            }
            else
            {
                tar->next = right;
                right = right->next;
            }
            tar = tar->next;
        }
        if (left) tar->next = left;
        if (right) tar->next = right;
        return dummy_node.next;
    }
};

4. 复杂度分析

空间复杂度

首先分析空间复杂度,可以看到全程只是建立了一个大小为64的指针数组以及归并排序过程中创建了一个伪头节点,因此空间复杂度为O(1)

时间复杂度

时间复杂度的分析就稍有些复杂,不妨从指针数组中每层发生的归并的次数来看:

在这里插入图片描述

在数组第0位上,需要进行n/2次归并得到n/2条的结点数为2的有序链表;在数组第1位上,需要进行n/4次归并,得到n/4条结点数位4的有序链表,依次类推。

在每层的归并操作中,执行比较总次数都是O(n),而总层数可以用O(logn)表示,因此总的时间复杂度就是O(nlogn)。

学习参考

学习更多相关知识请参考零声 github。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2236164.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【网络-交换机】生成树协议、环路检测

路由优先级 路由优先级决定了在多种可达的路由类型中&#xff0c;哪种路由将被用来转发数据包。路由优先级值越低&#xff0c;对应路由的优先级越高&#xff0c;优先级值255表示对应的路由不可达。一般情况下&#xff0c;静态路由的优先级为1&#xff0c;OSPF路由优先级为110&a…

确定图像的熵和各向异性 Halcon entropy_gray 解析

1、图像的熵 1.1 介绍 图像熵&#xff08;image entropy&#xff09;是图像“繁忙”程度的估计值&#xff0c;它表示为图像灰度级集合的比特平均数&#xff0c;单位比特/像素&#xff0c;也描述了图像信源的平均信息量。熵指的是体系的混乱程度&#xff0c;对于图像而言&#…

数字后端零基础入门系列 | Innovus零基础LAB学习Day9

Module 16 Wire Editing 这个章节的学习目标是学习如何在innovus中手工画线&#xff0c;切断一根线&#xff0c;换孔&#xff0c;更改一条net shape的layer和width等等。这个技能是每个数字IC后端工程师必须具备的。因为项目后期都需要这些技能来修复DRC和做一些手工custom走线…

除草机器人算法以及技术详解!

算法详解 图像识别与目标检测算法 Yolo算法&#xff1a;这是目标检测领域的一种常用算法&#xff0c;通过卷积神经网络对输入图像进行处理&#xff0c;将图像划分为多个网格&#xff0c;每个网格生成预测框&#xff0c;并通过非极大值抑制&#xff08;NMS&#xff09;筛选出最…

ProCalun卡伦纯天然万用膏,全家的皮肤健康守护

受季节交替、生活环境变化、空气污染等方面因素的影响&#xff0c;加上作息不规律导致的免疫力降低&#xff0c;我们或多或少会出现一些如湿疹、痤疮、瘙痒之类的皮肤问题&#xff0c;且反复概率很高。很多人盲目用药&#xff0c;甚至诱发激素依赖性皮炎。所以近年来&#xff0…

Vue 自定义icon组件封装SVG图标

通过自定义子组件CustomIcon.vue使用SVG图标&#xff0c;相比iconfont下载文件、重新替换更节省时间。 子组件包括&#xff1a; 1. Icons.vue 存放所有SVG图标的path 2. CustomIcon.vue 通过icon的id索引对应的图标 使用的时候需要将 <Icons></Icons> 引到使用的…

wireshark工具使用

复制数据 1.右键展开整帧数据 2.复制“所有可见项目” mark标记数据 标记&#xff1a; 跳转&#xff1a; 保存成文件&#xff1a; 文件–>导出特定分组—>Marked packets only

【SpringCloud】SpringBoot集成Swagger 常用Swagger注解

概述&#xff1a;SpringBoot集成Swagger 常用Swagger注解 导语 相信无论是前端还是后端开发&#xff0c;都或多或少地被接口文档折磨过。前端经常抱怨后端给的接口文档与实际情况不一致。后端又觉得编写及维护接口文档会耗费不少精力&#xff0c;经常来不及更新。其实无论是前…

Webserver(4.9)本地套接字的通信

目录 本地套接字 本地套接字 TCP\UDP实现不同主机、网络通信 本地套接字实现本地的进程间的通信&#xff0c;类似的&#xff0c;一般采用TCP的通信流程 生成套接字文件 #include<arpa/inet.h> #include<stdio.h> #include<stdlib.h> #include<unistd.h&…

[spring源码]spring配置类解析

解析配置类 在启动Spring时&#xff0c;需要传入一个AppConfig.class给ApplicationContext&#xff0c;ApplicationContext会根据AppConfig类封装为一个BeanDefinition&#xff0c;这种BeanDefinition我们把它称为配置类BeanDefinition AnnotationConfigApplicationContext a…

uni-app跨域set-cookie

set-cookie的值是作为一个权限控制的 首先&#xff0c;无论什么接口都会返回一个set-cookie&#xff0c;但未登录时&#xff0c;set-cookie是没有任何权限的 其次&#xff0c;登录接口请求时会修改set-cookie&#xff0c;并且在后续其他接口发起请求时&#xff0c;会在请求头…

chrdevbase驱动之Makefile优化(指定路径复制、删除文件)

对于学习嵌入式linux驱动篇的chrdevbase虚拟设备驱动时&#xff0c;需要将chrdevbase.c编译成.ko文件&#xff0c;应用层程序里需要把chrdevbaseAPP编译成chrdevbaseAPP可执行文件&#xff0c;此外还需要将生成的*.ko *APP文件拷贝至指定目录下&#xff0c;每次修改或者编译代码…

kafka实时返回浏览数据

在安装完kafka(Docker安装kafka_docker 部署kafka-CSDN博客)&#xff0c;查看容器是否启动&#xff1a; docker ps | grep -E kafka|zookeeper 再用python开启服务 from fastapi import FastAPI, Request from kafka import KafkaProducer import kafka import json import …

使用QtWebEngine的Mac应用如何发布App Store

前言 因为QtWebEngine时第三方包,苹果并不直接支持进行App Store上签名和发布,所以构建和发布一个基于使用QtWebEngine的应用程序并不容易,这里我们对Qt 5.8稍微做一些修改,以便让我们的基于QtWeb引擎的应用程序并让签名能够得到苹果的许可。 QtWebEngine提供了C++和Qml的…

C++虚继承演示

在继承中如果出现&#xff1a; 这种情况&#xff0c;B和C都继承了A&#xff0c;D继承了B、C 在D中访问A的成员会出现&#xff1a; 这样的警告 是因为在继承时A出现两条分支&#xff1a;ABD、ACD 编译器不知道访问的A中的元素是经过B继承还是C继承 所以B、C在继承A时要用到…

【赵渝强老师】Redis的RDB数据持久化

Redis 是内存数据库&#xff0c;如果不将内存中的数据库状态保存到磁盘&#xff0c;那么一旦服务器进程退出会造成服务器中的数据库状态也会消失。所以 Redis 提供了数据持久化功能。Redis支持两种方式的持久化&#xff0c;一种是RDB方式&#xff1b;另一种是AOF&#xff08;ap…

【计算机网络】章节 知识点总结

一、计算机网络概述 1. 计算机网络向用户提供的两个最重要的功能&#xff1a;连通性、共享 2. 因特网发展的三个阶段&#xff1a; 第一阶段&#xff1a;从单个网络 ARPANET 向互联网发展的过程。1983 年 TCP/IP 协议成为 ARPANET 上的标准协议。第二阶段&#xff1a;建成三级…

Linux基础(十)——磁盘分区、格式化、检验和挂载

磁盘分区、格式化、检验和挂载 1.观察磁盘的分区状态2.UUID3.磁盘分区&#xff08;gdisk/fdisk&#xff09;3.1 gdisk3.2 fdisk 4.磁盘的格式化4.1 XFS文件系统的格式化4.2 ext4文件系统的格式化 5.文件系统的救援6.文件系统的挂载与卸载6.1 挂载6.2 卸载 7.设置开机挂载8.特殊…

Android的BroadcastReceiver

1.基本概念&#xff1a;BroadCast用于进程间或者线程间通信 本质上是用Binder方法&#xff0c;以AMS为订阅中心&#xff0c;完成注册&#xff0c;发布&#xff0c;监听的操作。 2.简单实现的例子 package com.android.car.myapplication;import android.content.BroadcastRe…

一招解决Mac没有剪切板历史记录的问题

使用Mac的朋友肯定都为Mac的剪切功能苦恼过&#xff0c;旧内容覆盖新内容&#xff0c;导致如果有内容需要重复输入的话&#xff0c;就需要一次一次的重复复制粘贴&#xff0c;非常麻烦 但其实Mac也能够有剪切板历史记录功能&#xff0c;iCopy&#xff0c;让你的Mac也能拥有剪切…