服务器之家:专注于VPS、云服务器配置技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - C/C++ - 学习在 C++ 中将合并排序算法与链表一起使用

学习在 C++ 中将合并排序算法与链表一起使用

2023-12-15 15:44鲨鱼编程 C/C++

本文介绍了如何在C++中将合并排序算法与链表一起使用,实现链表的轻松排序。

一、引言

链表是一种常见的数据结构,用于存储一系列有序或无序的元素。在实际应用中,我们经常需要对链表进行排序。合并排序(Merge Sort)是一种高效的排序算法,具有稳定的排序性能和O(nlogn)的时间复杂度。本文将介绍如何在C++中将合并排序算法与链表一起使用,以便轻松实现链表的排序。

学习在 C++ 中将合并排序算法与链表一起使用

二、链表基础

链表是一种通过指针链接在一起的数据结构。每个节点包含数据和指向下一个节点的指针。在C++中,我们可以定义一个结构体来表示链表节点,如下所示:

struct ListNode {  

    int val;           // 节点值  

    ListNode* next;    // 指向下一个节点的指针  

    ListNode(int x) : val(x), next(nullptr) {} // 构造函数  

};

三、合并排序算法原理

合并排序算法采用分治策略,将待排序数组不断拆分为子数组,直到子数组长度为1或0,然后将子数组两两合并,直到最终得到有序数组。合并排序算法的时间复杂度为O(nlogn),是一种稳定的排序算法。

四、合并排序与链表的结合

将合并排序算法应用于链表时,我们需要对链表进行拆分和合并操作。拆分操作可以通过找到链表的中间节点实现,合并操作则需要比较两个链表节点的值并进行链接。具体实现如下:

1.链表拆分

为了找到链表的中间节点,我们可以使用快慢指针法。定义两个指针slow和fast,初始时都指向链表的头节点。然后,slow每次移动一步,fast每次移动两步。当fast到达链表末尾时,slow刚好到达链表中间。代码如下:

ListNode* GetMiddleNode(ListNode* head) {    
    if (head == nullptr || head->next == nullptr) {  
        return head; // 对于空链表或只有一个节点的链表,返回头节点  
    }  
      
    ListNode* slow = head;    
    ListNode* fast = head;    
    while (fast != nullptr && fast->next != nullptr && fast->next->next != nullptr) {    
        slow = slow->next;    
        fast = fast->next->next;    
    }    
    return slow;    
}

2.链表合并

合并两个链表时,我们需要定义一个新的头节点dummy,用于连接合并后的链表。然后,比较两个链表的节点值,将较小的节点链接到新链表的末尾。重复此过程,直到其中一个链表为空。最后,将非空链表的剩余部分链接到新链表的末尾。代码如下:

ListNode* MergeLists(ListNode* list1, ListNode* list2) {  
    ListNode dummy(0);  
    ListNode* tail = &dummy;  
  
    while (list1 && list2) {  
        if (list1->val < list2->val) {  
            tail->next = list1;  
            list1 = list1->next;  
        } else {  
            tail->next = list2;  
            list2 = list2->next;  
        }  
        tail = tail->next;  
    }  
  
    tail->next = (list1 != nullptr) ? list1 : list2;  
    return dummy.next;  
}

3.合并排序链表实现

结合以上两个步骤,我们可以实现链表的合并排序。首先,递归地将链表拆分为子链表,直到子链表长度为1或0。然后,将子链表两两合并,直到最终得到有序链表。代码如下:

ListNode* MergeSortList(ListNode* head) {  
    if (head == nullptr || head->next == nullptr) {  
        return head; // 链表为空或只有一个节点,无需排序  
    }  
    ListNode* middle = GetMiddleNode(head); // 找到链表中间节点  
    ListNode* nextToMiddle = middle->next; // 记录中间节点的下一个节点  
    middle->next = nullptr; // 将链表拆分为两个子链表  
    return MergeLists(MergeSortList(head), MergeSortList(nextToMiddle)); // 递归地对子链表进行排序并合并结果  
}

五、性能分析

合并排序算法的时间复杂度为O(nlogn),其中n为链表的长度。在空间复杂度方面,由于递归实现需要使用栈空间,因此空间复杂度为O(logn)。这使得合并排序算法在处理大规模数据时具有较高的效率。

六、应用示例

下面是一个简单的示例,展示如何使用合并排序算法对链表进行排序:

#include < iostream>  
  
void PrintList(ListNode* head) {  
    while (head != nullptr) {  
        std::cout << head->val << " ";  
        head = head->next;  
    }  
    std::cout << std::endl;  
}  
  
int main() {  
    // 创建一个无序链表: 4 -> 2 -> 1 -> 3  
    ListNode* head = new ListNode(4);  
    head->next = new ListNode(2);  
    head->next->next = new ListNode(1);  
    head->next->next->next = new ListNode(3);  
  
    std::cout << "Before sorting:" << std::endl;  
    PrintList(head); // 输出: 4 2 1 3  
  
    head = MergeSortList(head); // 对链表进行排序  
  
    std::cout << "After sorting:" << std::endl;  
    PrintList(head); // 输出: 1 2 3 4  
  
    // 释放链表内存  
    while (head != nullptr) {  
        ListNode* temp = head;  
        head = head->next;  
        delete temp;  
    }  
  
    return 0;  
}

七、总结

本文介绍了如何在C++中将合并排序算法与链表一起使用,实现链表的轻松排序。通过详细讲解链表基础、合并排序算法原理以及具体实现步骤,希望能够帮助读者更好地理解和掌握这一技术。合并排序算法在处理大规模数据时具有较高的效率,因此在实际应用中具有广泛的应用前景。

原文地址:https://mp.weixin.qq.com/s?__biz=Mzg4Mjg0MTA3Ng==&mid=2247487114&idx=1&sn=822d176e5af8061b71a6d5c033fb9130

延伸 · 阅读

精彩推荐
  • C/C++C++ 类和对象基础篇

    C++ 类和对象基础篇

    类是创建对象的模板,一个类可以创建多个对象,每个对象都是类类型的一个变量;创建对象的过程也叫类的实例化。每个对象都是类的一个具体实例(...

    夜行过客8352021-08-12
  • C/C++C语言求逆矩阵案例详解

    C语言求逆矩阵案例详解

    这篇文章主要介绍了C语言求逆矩阵案例详解,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下...

    dogdng5852021-12-16
  • C/C++C语言关于自定义数据类型之枚举和联合体详解

    C语言关于自定义数据类型之枚举和联合体详解

    枚举顾名思义就是把所有的可能性列举出来,像一个星期分为七天我们就可以使用枚举,联合体是由关键字union和标签定义的,和枚举是一样的定义方式,...

    Ersansui10192022-02-28
  • C/C++Windows的钩子机制详解

    Windows的钩子机制详解

    这篇文章主要介绍了Windows的钩子机制,对于初学者进一步了解windows程序设计中钩子的原理及运用有很大的帮助,需要的朋友可以参考下...

    C语言程序设计4842021-01-22
  • C/C++C++中静态成员函数访问非静态成员的实例

    C++中静态成员函数访问非静态成员的实例

    这篇文章主要介绍了C++中静态成员函数访问非静态成员的实例的相关资料,需要的朋友可以参考下...

    shimazhuge6062021-05-21
  • C/C++C++11中的chrono库详解

    C++11中的chrono库详解

    C++11提供了日期时间相关的库chrono,通过chrono库可以很方便的处理日期和时间,这篇文章主要介绍了C++11中的chrono库,需要的朋友可以参考下...

    你好,冯同学5242023-03-19
  • C/C++VSCODE+cmake配置C++开发环境的实现步骤

    VSCODE+cmake配置C++开发环境的实现步骤

    这篇文章主要介绍了VSCODE+cmake配置C++开发环境的实现步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋...

    ha_____ha9632021-10-27
  • C/C++C++中string字符串分割函数split()的4种实现方法

    C++中string字符串分割函数split()的4种实现方法

    最近笔试经常遇到需要对字符串进行快速分割的情景,下面这篇文章主要给大家介绍了关于C++中string字符串分割函数split()的4种实现方法,文中通过实例代码介...

    我行我素,向往自由4662022-12-26