【数据结构】单链表的定义和操作

目录

1.单链表的定义

2.单链表的创建和初始化

3.单链表的插入节点操作

4.单链表的删除节点操作

5.单链表的查找节点操作

6.单链表的更新节点操作

7.完整代码


🌈嗨!我是Filotimo__🌈。很高兴与大家相识,希望我的博客能对你有所帮助。

💡本文由Filotimo__✍️原创,首发于CSDN📚。

📣如需转载,请事先与我联系以获得授权⚠️。

🎁欢迎大家给我点赞👍、收藏⭐️,并在留言区📝与我互动,这些都是我前进的动力!

🌟我的格言:森林草木都有自己认为对的角度🌟。

1.单链表的定义

单链表由一系列节点组成,每个节点包含数据和一个指向下一个节点的指针(通常称为"next"指针)。单链表的最后一个节点指向空值(null)表示链表的结束。

单链表通常包含以下几个要素:

节点(Node):表示链表中的每个元素,通常包含数据和一个指向下一个节点的指针。
头结点(Head Node):代表链表的开始,通常不存储数据,它的next指针指向第一个实际节点。
尾节点(Tail Node):代表链表的最后一个节点,它的next指针指向空值(null)。
链表的长度:即链表中节点的数量。

头节点是第一个节点,尾节点是最后一个节点,它们都不存储数据元素。

单链表的特点是可以实现动态的插入和删除操作,但是访问某个节点时需要从头结点开始依次遍历链表,效率较低。

2.单链表的创建和初始化

创建单链表需要定义一个节点结构,该结构包含一个数据域和一个指向下一个节点的指针。首先创建一个头节点,将其指针设置为NULL,表示链表为空。

struct ListNode {
    int data;
    ListNode* next;
};

ListNode* createLinkedList() {
    ListNode* head = new ListNode();
    head->next = NULL;
    return head;
}

先定义名为 ListNode 的结构体。该结构体包含两个成员变量:int data 和 ListNode* next。

int data 用来存储节点的数据值,表示链表中的一个元素。

ListNode* next 是指向下一个节点的指针,表示链表中当前节点的下一个节点。

在函数 createLinkedList中,首先通过 new ListNode() 创建了一个新的节点对象,并将其地址赋值给 head 指针变量。然后,将 head->next 设置为 NULL,表示链表当前只有一个节点,并且没有下一个节点。最后,将 head 返回作为链表的头节点指针。

3.单链表的插入节点操作

在单链表中插入一个新节点需要先找到插入位置的前一个节点,然后将新节点插入到其后面。

void insertNode(ListNode* list, int value) {
    ListNode* newNode = new ListNode();
    newNode->data = value;
    newNode->next = NULL;

    ListNode* p = list;
    while (p->next != NULL) {
        p = p->next;
    }

    p->next = newNode;
}

ListNode* list表示链表的头节点指针

int value表示要插入节点的数值

函数中,先通过 `new ListNode()` 创建了一个新的节点对象,并将其地址赋给 `newNode` 指针变量。然后,将 `value` 的值赋给 `newNode->data`,即将要插入节点的数值存储到节点的 `data` 成员变量中。接着,将 `newNode->next` 设置为 `NULL`,表示新节点的下一个节点暂时为空。

再通过创建一个指针变量 `p`,将其初始化为 `list`,即将其指向链表的头节点。使用循环遍历链表,直到找到最后一个节点(即 `p->next = NULL`)。在每次循环迭代中,通过 `p = p->next` 将指针 `p` 移动到下一个节点。

在循环结束后,将新节点 `newNode` 插入到链表的最后一个节点的后面,即将最后一个节点的 `next` 指针指向 `newNode`。

4.单链表的删除节点操作

删除单链表中的一个节点也需要找到待删除节点的前一个节点,将其指针指向待删除节点的下一个节点,然后释放待删除节点的内存。

void deleteNode(ListNode* list, int value) {
    ListNode* p = list;
    while (p->next != NULL) {
        if (p->next->data == value) {
            ListNode* temp = p->next;
            p->next = temp->next;
            delete temp;
            break;
        }
        p = p->next;
    }
}

函数中,首先创建一个指针变量p,将其初始化为list,即将其指向链表的头节点。然后,使用循环遍历链表,直到找到节点数值等于value的节点,或者遍历到链表的最后一个节点。

在每次循环迭代中,通过判断p->next->data是否等于value,确定是否找到了需要删除的节点。如果找到了目标节点,则创建一个临时指针变量 `temp`,将其指向待删除节点的地址。通过 `p->next = temp->next` 更新前一个节点的 `next` 指针,跳过待删除节点。使用 `delete temp` 释放内存,删除待删除节点,并通过 `break` 语句跳出循环。

5.单链表的查找节点操作

查找单链表中某个特定值的节点,需要遍历链表,逐个比较节点的数据域。

ListNode* findNode(ListNode* list, int value) {
    ListNode* p = list;
    while (p->next != NULL) {
        if (p->next->data == value) {
            return p->next;
        }
        p = p->next;
    }
    return NULL;  // 未找到返回NULL
}

这个函数,使用循环遍历链表,直到找到节点数值等于value的节点,或者遍历到链表的最后一个节点。在每次循环迭代中,通过判断p->next->data是否等于value,确定是否找到了需要查找的节点。如果找到了目标节点,则直接返回该节点的指针p->next。

6.单链表的更新节点操作

更新单链表中某个节点的值,需要先找到该节点,然后修改其数据域的值。

void updateNode(ListNode* list, int oldValue, int newValue) {
    ListNode* p = findNode(list, oldValue);
    if (p != NULL) {
        p->data = newValue;
    }
}

函数的参数:

ListNode* list表示链表的头节点指针

oldValue表示待更新节点的旧数值

newValue表示待更新节点的新数值

在函数中,首先通过调用indNode函数来查找链表中数值为oldValue的节点,将返回的节点指针赋值给p。

接着,通过判断p是否为NULL,即是否找到了待更新的节点。如果p != NULL,则说明找到了需要更新的节点,将该节点的data成员变量的值更新为newValue,即p->data = newValue。

7.完整代码

#include <iostream>

struct ListNode {
    int data;
    ListNode* next;
};

ListNode* createLinkedList() {
    ListNode* head = new ListNode();
    head->next = NULL;
    return head;
}

void insertNode(ListNode* list, int value) {
    ListNode* newNode = new ListNode();
    newNode->data = value;
    newNode->next = NULL;

    ListNode* p = list;
    while (p->next != NULL) {
        p = p->next;
    }

    p->next = newNode;
}

void deleteNode(ListNode* list, int value) {
    ListNode* p = list;
    while (p->next != NULL) {
        if (p->next->data == value) {
            ListNode* temp = p->next;
            p->next = temp->next;
            delete temp;
            break;
        }
        p = p->next;
    }
}

ListNode* findNode(ListNode* list, int value) {
    ListNode* p = list;
    while (p->next != NULL) {
        if (p->next->data == value) {
            return p->next;
        }
        p = p->next;
    }
    return NULL;
}

void updateNode(ListNode* list, int oldValue, int newValue) {
    ListNode* p = findNode(list, oldValue);
    if (p != NULL) {
        p->data = newValue;
    }
}

void printLinkedList(ListNode* list) {
    ListNode* p = list->next;
    while (p != NULL) {
        std::cout << p->data << " ";
        p = p->next;
    }
    std::cout << std::endl;
}

int main() {
    ListNode* list = createLinkedList();

    // 插入节点
    insertNode(list, 1);
    insertNode(list, 2);
    insertNode(list, 3);
    insertNode(list, 4);
    std::cout << "链表:";
    printLinkedList(list);

    // 删除节点
    deleteNode(list, 2);
    std::cout << "删除节点2后的链表:";
    printLinkedList(list);

    // 查找节点
    ListNode* node = findNode(list, 3);
    if (node != NULL) {
        std::cout << "找到节点3" << std::endl;
    } else {
        std::cout << "未找到节点3" << std::endl;
    }

    // 更新节点
    updateNode(list, 3, 5);
    std::cout << "更新节点3的值为5后的链表:";
    printLinkedList(list);

    return 0;
}

输出结果如下:

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

汽车充电协议OpenV2G的平替cbexigen!!

纵所周知&#xff0c;开源欧规协议 CCS 的 OpenV2G 协议不支持 ISO15118-20:2022 协议&#xff0c;并且软件维护者已经明确不在进行该软件的维护。 前几天在 Github 上冲浪发现了一个宝藏开源项目&#xff0c;完美的实现的 OpenV2G 的 Exidizer 工具的功能&#xff1a;cbexigen…

Goldstein枝切法对存在间断相位缺陷的解缠研究

摘要: Goldstein枝切法作为相位解缠中路径积分法的重要算法之一&#xff0c;其解缠结果易受到噪声或间断相位缺陷所引起的残差点影响。为了研究相位间断缺陷对解缠算法的影响&#xff0c;模拟了具有间断相位缺陷的数据&#xff0c;采用 Gold-stein枝切法进行了系统的解缠研究。…

嵌入式C语言变量、数组、指针初始化的多种操作

在敲代码的时候&#xff0c;我们会给变量一个初始值&#xff0c;以防止因为编译器的原因造成变量初始值的不确定性。 对于数值类型的变量往往初始化为0&#xff0c;但对于其他类型的变量&#xff0c;如字符型、指针型等变量等该如何初始化呢&#xff1f; 数值类变量初始化 整…

巴贝拉葡萄酒是单一品种还是混合品种制成的?

大多数巴贝拉葡萄酒都是由单一的巴贝拉葡萄品种制成的&#xff0c;许多意大利葡萄酒商开始尝试在巴贝拉葡萄酒中加入其它葡萄品种&#xff0c;其中两个最受欢迎的意大利品种是皮埃蒙特的巴贝拉德阿尔巴和达斯蒂。和朋友在一家意大利餐厅吃饭&#xff0c;被酒单吓到了&#xff1…

MySQL数据库 入门

目录 一、MySQL概述 二、MySQL安装 安装数据库 配置数据库 启动停止数据库 客户端连接数据库 三、数据模型 四、SQL 一、MySQL概述 先来讲解三个概念&#xff1a;数据库、数据库管理系统、 SQL 。 而目前主流的关系型数据库管理系统的市场占有率排名如下&#xff1a; …

Android--Jetpack--数据库Room详解一

人生何须万种愁&#xff0c;千里云烟一笑收 一&#xff0c;定义 Room也是一个ORM框架&#xff0c;它在SQLite上提供了一个抽象层&#xff0c;屏蔽了部分底层的细节&#xff0c;使用对象对数据库进行操作&#xff0c;进行CRUD就像对象调用方法一样的简单。 二&#xff0c;角色介…

​【EI会议征稿中】#先投稿,先送审#第三届网络安全、人工智能与数字经济国际学术会议(CSAIDE 2024)​

#先投稿&#xff0c;先送审# 第三届网络安全、人工智能与数字经济国际学术会议&#xff08;CSAIDE 2024&#xff09; 2024 3rd International Conference on Cyber Security, Artificial Intelligence and Digital Economy 2024年3月1日-3日 | 中国南京 会议官网&#xff1a…

外包干了4个月,测试技术退步明显

先说一下自己的情况&#xff0c;本科生&#xff0c;20年通过校招进入杭州某软件公司&#xff0c;干了3年的功能测试&#xff0c;当然有半年是被封在了家里&#xff0c;今年年初&#xff0c;感觉自己不能够在这样下去了&#xff0c;长时间呆在一个舒适的环境会让一个人堕落!而我…

10个 Python 脚本来自动化你的日常任务

在这个自动化时代&#xff0c;我们有很多重复无聊的工作要做。想想这些你不再需要一次又一次地做的无聊的事情&#xff0c;让它自动化&#xff0c;让你的生活更轻松。 那么在本文中&#xff0c;我将向您介绍 10 个 Python 自动化脚本&#xff0c;以使你的工作更加自动化&#…

在Mac系统下为SpringBoot 配置Maven 【避坑完整版】

前提背景 电脑罢工&#xff0c;操作系统重装&#xff0c;不仅有大量的软件需要安装&#xff0c;还有很多开发环境需要配置。 就在今天配置Maven的时候各种坑&#xff0c;写下来供大家参考。 一、在讨论安装Maven前先安装一下JDK,方式很多&#xff0c;我这里有个比较快的办法&am…

gitee(ssh)同步本地

一、什么是码云 gitee Git的”廉价平替” > 服务器在国内&#xff0c;运行不费劲 在国内也形成了一定的规模 git上的一些项目插件等在码云上也可以找得到 二、创建仓库 三、删除仓库 四、仓库与本地同步 > 建立公钥 五、把仓库同步到本地 六、在本地仓库中创建vue项目…

Docker 部署 Lobe Chat 服务

拉取最新版本的 Lobe Chat 镜像&#xff1a; $ sudo docker pull lobehub/lobe-chat:latest使用以下命令来运行 Lobe Chat 容器: $ sudo docker run -d --name lobe-chat -p 10084:3210 -e OPENAI_API_KEYsk-xxxx -e OPENAI_PROXY_URLhttps://api.openai.com/v1 -e ACCESS_CO…

C之switch小问题

执行结果&#xff1a; 为什么会是100呢&#xff1f; 因为C语言会忽视 switch语句与第一个case之间的code&#xff0c;也就是根本不会执行 “num100;

谷歌Gemini API 应用(二):LangChain 加持

昨天我完成了谷歌Gemini API 应用(一)&#xff1a;基础应用这篇博客&#xff0c;今天我们要在此基础上实现Gemini模型的Langchian加持&#xff0c;因为Gemini API刚发布没几天&#xff0c;所以langchian还没有来得及将其整合到现有的langchain包的架构内&#xff0c;langchain公…

【SpringBoot篇】Interceptor拦截器 | 拦截器和过滤器的区别

文章目录 &#x1f339;概念⭐作用 &#x1f384;快速入门⭐入门案例代码实现 &#x1f6f8;拦截路径&#x1f354;拦截器interceptor和过滤器filter的区别&#x1f386;登录校验 &#x1f339;概念 拦截器&#xff08;Interceptor&#xff09;是一种软件设计模式&#xff0c;…

Docker构建镜像时空间不足:/var/lib/docker,no space left on device

背景 在一次更新业务服务功能后&#xff0c;重新在服务器上构建微服务镜像&#xff0c;在构建镜像时报错空间不足&#xff1a; /var/lib/docker, no space left on device 赶紧用 df -h 看了下磁盘使用情况&#xff0c;果然&#xff0c; devicemapper 已经满了。。由于需要紧急…

C语言代码实现URL编码

在 Python&#xff0c;只需要导入 urllib.parse&#xff0c;然后使用 quote 函数即可把任意字符串进行 URL 编码 现在使用 C 语言来实现等效的代码&#xff0c;我在网上找到现成的代码&#xff0c;改代码接收命令行输入参数&#xff0c;然后进行 URL 编码并输出&#xff1a; #…

Flask基本用法:一个HelloWorld,搭建服务、发起请求

目录 1、简介 2、安装 3、Flask使用示例 参考 1、简介 官网文档 Flask是一个轻量的web服务框架&#xff0c;我们可以利用它快速搭建一个服务&#xff0c;对外提供接口&#xff0c;其他人可以轻松调用我们的服务。这对算法工程师来说比较关键&#xff0c;我们通常不擅长搞开发…

【Docker四】使用Docker-compose一键部署Wordpress平台

目录 一、YAML 文件格式及编写注意事项&#xff08;重要&#xff09; 1、yaml文件使用时注意事项&#xff1a; 2、yaml文件的基本数据结构&#xff1a; 2.1、声明变量&#xff08;标量。是单个的不可再分的值&#xff0c;类型&#xff1a;字符串&#xff0c;整数&#xff0c…