使用C++实现单链表的操作与实践

2025-02-10 16:50

本文主要是介绍使用C++实现单链表的操作与实践,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

《使用C++实现单链表的操作与实践》在程序设计中,链表是一种常见的数据结构,特别是在动态数据管理、频繁插入和删除元素的场景中,链表相比于数组,具有更高的灵活性和高效性,尤其是在需要频繁修改数据结构的应...

一、单链表的基本概念

单链表是一种由节点组成的线性数据结构,其中每个节点包含数据部分和指向下一个节点的指针。与数组不同,链表的节点在内存中不要求连续存储,而是通过指针连接。因此,链表的插入和删除操作较为灵活,不需要大量的数据移动。

在C++中,我们通过类的封装特性来实现面向对象的链表,这不仅能有效管理链表的内存,还能通过封装实现更易用、更安全的操作。

二、单链表类的设计

我们将通过一个简单的C++类来实现单链表,该类包含基本的链表操作,如插入、删除、打印链表等。

1. 节点的定义

首先,我们定义了一个 Node 结构体来表示链表中的每个节点。每个节点包含一个数据部分 data 和一个指向下一个节点的指针&n编程bsp;next

struct Node {
    int data;      // 数据域
    Node* next;    // 指针域,指向下一个节点
};

2. 链表的类定义

接下来,我们定义 List 类,它包含一个指向链表头部的指针 phead,以及若干成员函数来实现链表的常见操作。

class List {
private:
    Node* phead; // 链表头指针

public:
    // 构造函数
    List() : phead(nullptr) {}

    // 析构函数
    ~List() {
        while (phead != nullptr) {
            PopFront();
        }
    }

    // 创建节点
    Node* CreateNode(int x) {
        Node* node = new Nodandroide;
        node->data = x;
        node->next = nullptr;
        return node;
    }

    // 打印链表
    void PrintList() {
        Node* cur = phead;
        while (cur) {
            cout << cur->data << "-->";
            cur = cur->next;
        }
        cout << "NULL" << endl;
    }

    //eSeNtxKJV 头插法
    void PushFront(int x) {
        Node* newNode = CreateNode(x);
        newNode->next = phead;
        phead = newNode;
    }

    // 尾插法
    void PushBack(int x) {
        Node* newNode = CreateNode(x);
        if (phead == nullptr)
            phead = newNode;
        else {
            Node* tail = phead;
            while (tail->next != nullptr) {
                tail = tail->next;
            }
            tail->next = newNode;
        }
    }

    // 头删
    void PopFront() {
        if (phead == nullptr)
            cout << "链表为空,无法进行删除操作!" << endl;
        else {
            Node* del = phead;
            phead = del->next;
            delete del;
            del = nullptr;
        }
    }

    // 尾删
    void PopBack() {
        if (phead == nullptr)
            cout << "链表为空,无法进行删除操作!" << endl;
        else {
            if (phead->next == nullptr) {
                delete phead;
                phead = nullptr;
            } else {
                Node* tail = phead;
                while (tail->next->next != nullptr) {
                    tail = tail->next;
                }
                delhttp://www.chinasem.cnete tail->next;
                tail->next = nullptr;
            }
        }
    }
};

三、单链表的操作实现

  • PushFront: 在链表的头部插入新节点。
  • PushBack: 在链表的尾部插入新节点。
  • PopFront: 删除链表的头节点。
  • PopBack: 删除链表的尾节点。
  • PrintList: 打印链表中的所有节点。

四、测试与演示

下面的 main 函数展示了如何使用上述链表类实现基本操作:

int main() {
    List ls1;  // 创建一个链表对象

    // 进行一些操作
    ls1.PushBack(1);
    ls1.PushBack(2);
    ls1.PushBack(3);
    ls1.PushBack(4);
    ls1.PushBack(5);

    // 打印链表
    ls1.PrintList();

    // 头删除和尾删除
    ls1.PopFront();
    ls1.PopBack();

    // 头插操作
    ls1.PushFront(9);

    // 打印链表
    ls1.PrintList();

    return 0;
}

五、链表操作的复杂度

  1. PushFront 和 PopFront:这两个操作的时间复杂度为 O(1),因为它们仅仅操作链表的头节点。
  2. PushBack 和 PopBack:这两个操作的时间复杂度为 O(n),需要遍历整个链表,直到找到尾节点。
  3. PrintList:打印链表的时间复杂度为 O(n),需要遍历所有节点。

六、完整代码

#include<IOStream>
using namespace std;
//节点类型声明
structpython Node
{
    int date;
    Node* next;
};
class List
{
private:
    //成员变量
    Node* phead;
public:
    //成员函数
    List() : phead(nullptr) {}//构造函数
    ~List()//析构函数
    {
        while(phead!=NULL)
        {
            PopFront();
        }
    }
    Node* CreateNode(int x)//创建节点
    {
        Node* node=new Node;
        node->date=x;
        node->next=NULL;
        return node;
    }
    void PrintList()//打印链表
    {
        Node *cur=phead;
        while(cur)
        {
            cout<<cur->date<<"-->";
            cur=cur->next;
        }
        cout<<"NULL"<<endl;
    }
    void PushFront(int x)//头插
    {
        Node*newnode=CreateNode(x);
        newnode->next=phead;
        phead=newnode;
    }
    void PushBack(int x)//尾插
    {
        Node*newnode=CreateNode(x);
        if(phead==NULL)
            phead=newnode;
        else
        {
            Node* tail = phead;
            while (tail->next != NULL)
            {
                tail = tail->next;
            }
            tail->next = newnode;
        }

    }
    void PopFront() //头删
    {
        if (phead==NULL)
            cout<<"链表为空,无法进行删除操作!"<<endl;
        else
        {
            Node* del=phead;
            phead=del->next;
            delete del;
            del=NULL;
        }
    }

    void PopBack()  //尾删
    {
        if (phead== NULL)
            cout<<"链表为空,无法进行删除操作!"<<endl;
       else
        {
           if(phead->next==NULL)
           {
               delete phead;
               phead=NULL;
           }
           else
           {
               Node* tail = phead;
               while (tail->next->next != NULL)
               {
                   tail = tail->next;
               }
               delete tail->next;
               tail->next=NULL;
           }
        }
    }

};
int main()
{
    List ls1;
    ls1.PushBack(1);
    ls1.PushBack(2);
    ls1.PushBack(3);
    ls1.PushBack(4);
    ls1.PushBack(5);
    ls1.PrintList();
    ls1.PopFront();
    ls1.PopBack();
    ls1.PushFront(9);
    ls1.PrintList();
    return 0;
}

七、总结

通过面向对象的方式实现单链表,我们可以更加方便和安全地进行链表操作。封装了节点管理、内存管理以及链表操作函数的类,让链表操作更加直观并且容易维护。在实际开发中,链表结构广泛应用于各种算法和数据管理系统,掌握链表的使用可以帮助我们高效地解决许多动态数据管理的问题。

以上就是使用C++实现单链表的操作与实践的详细内容,更多关于C++实现单链表的资料请关注China编程(www.chinasem.cn)其它相关文章!

这篇关于使用C++实现单链表的操作与实践的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1153377

相关文章

使用Java将各种数据写入Excel表格的操作示例

《使用Java将各种数据写入Excel表格的操作示例》在数据处理与管理领域,Excel凭借其强大的功能和广泛的应用,成为了数据存储与展示的重要工具,在Java开发过程中,常常需要将不同类型的数据,本文... 目录前言安装免费Java库1. 写入文本、或数值到 Excel单元格2. 写入数组到 Excel表格

redis中使用lua脚本的原理与基本使用详解

《redis中使用lua脚本的原理与基本使用详解》在Redis中使用Lua脚本可以实现原子性操作、减少网络开销以及提高执行效率,下面小编就来和大家详细介绍一下在redis中使用lua脚本的原理... 目录Redis 执行 Lua 脚本的原理基本使用方法使用EVAL命令执行 Lua 脚本使用EVALSHA命令

C#如何调用C++库

《C#如何调用C++库》:本文主要介绍C#如何调用C++库方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录方法一:使用P/Invoke1. 导出C++函数2. 定义P/Invoke签名3. 调用C++函数方法二:使用C++/CLI作为桥接1. 创建C++/CL

Python中pywin32 常用窗口操作的实现

《Python中pywin32常用窗口操作的实现》本文主要介绍了Python中pywin32常用窗口操作的实现,pywin32主要的作用是供Python开发者快速调用WindowsAPI的一个... 目录获取窗口句柄获取最前端窗口句柄获取指定坐标处的窗口根据窗口的完整标题匹配获取句柄根据窗口的类别匹配获取句

Java 中的 @SneakyThrows 注解使用方法(简化异常处理的利与弊)

《Java中的@SneakyThrows注解使用方法(简化异常处理的利与弊)》为了简化异常处理,Lombok提供了一个强大的注解@SneakyThrows,本文将详细介绍@SneakyThro... 目录1. @SneakyThrows 简介 1.1 什么是 Lombok?2. @SneakyThrows

在 Spring Boot 中实现异常处理最佳实践

《在SpringBoot中实现异常处理最佳实践》本文介绍如何在SpringBoot中实现异常处理,涵盖核心概念、实现方法、与先前查询的集成、性能分析、常见问题和最佳实践,感兴趣的朋友一起看看吧... 目录一、Spring Boot 异常处理的背景与核心概念1.1 为什么需要异常处理?1.2 Spring B

Python位移操作和位运算的实现示例

《Python位移操作和位运算的实现示例》本文主要介绍了Python位移操作和位运算的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录1. 位移操作1.1 左移操作 (<<)1.2 右移操作 (>>)注意事项:2. 位运算2.1

如何在 Spring Boot 中实现 FreeMarker 模板

《如何在SpringBoot中实现FreeMarker模板》FreeMarker是一种功能强大、轻量级的模板引擎,用于在Java应用中生成动态文本输出(如HTML、XML、邮件内容等),本文... 目录什么是 FreeMarker 模板?在 Spring Boot 中实现 FreeMarker 模板1. 环

Qt实现网络数据解析的方法总结

《Qt实现网络数据解析的方法总结》在Qt中解析网络数据通常涉及接收原始字节流,并将其转换为有意义的应用层数据,这篇文章为大家介绍了详细步骤和示例,感兴趣的小伙伴可以了解下... 目录1. 网络数据接收2. 缓冲区管理(处理粘包/拆包)3. 常见数据格式解析3.1 jsON解析3.2 XML解析3.3 自定义

使用Python和Pyecharts创建交互式地图

《使用Python和Pyecharts创建交互式地图》在数据可视化领域,创建交互式地图是一种强大的方式,可以使受众能够以引人入胜且信息丰富的方式探索地理数据,下面我们看看如何使用Python和Pyec... 目录简介Pyecharts 简介创建上海地图代码说明运行结果总结简介在数据可视化领域,创建交互式地