【数据结构】线性表(二)单链表及其基本操作(创建、插入、删除、修改、遍历打印)

news2025/1/19 2:57:57

目录

前文、线性表的定义及其基本操作(顺序表插入、删除、查找、修改)

四、线性表的链接存储结构

1. 单链表(C语言)

a. 链表节点结构

b. 创建新节点

c. 在链表末尾插入新节点

d. 删除指定节点

e. 修改指定节点的数据

f. 遍历链表并打印

g. 主函数

C语言代码整合

Cpp代码整合


前文、线性表的定义及其基本操作(顺序表插入、删除、查找、修改)

【数据结构】线性表(一)线性表的定义及其基本操作(顺序表插入、删除、查找、修改)-CSDN博客icon-default.png?t=N7T8https://blog.csdn.net/m0_63834988/article/details/132089038?spm=1001.2014.3001.5501

四、线性表的链接存储结构

        顺序表的优点是存取速度快。但是,无论是插入一个结点,还是删除一个结点,都需要调整一批结点的地址。要克服该缺点,就必须给出一种不同于顺序存储的存储方式。用链接存储方式存储的线性表被称为链表,可以克服上述缺点。

        链表中的结点用存储单元(若干个连续字节)来存放,存储单元之间既可以是(存储空间上)连续的,也可以是不连续的,甚至可以零散地分布在存储空间中的任何位置。换言之,链表中结点的逻辑次序和物理次序之间并无必然联系。最重要的是,链表可以在不移动结点位置的前提下根据需要随时添加删除结点,动态调整。

1. 单链表(C语言)

        在链接存储结构中,插入和删除操作相对于顺序存储结构而言更加高效,时间复杂度为O(1)。而查找操作的时间复杂度为O(n)。

a. 链表节点结构

typedef struct Node {
    int data;
    struct Node* next;
} Node;

        链表节点的结构体 Node,包含一个整数数据 data 和一个指向下一个节点的指针 next

b. 创建新节点

Node* createNode(int data) {
    Node* newNode = (Node*)malloc(sizeof(Node));
    if (newNode != NULL) {
        newNode->data = data;
        newNode->next = NULL;
    }
    return newNode;
}
  • 创建一个新的节点并返回指向该节点的指针:
    • 使用 malloc 分配了节点的内存空间;
    • 将传入的数据赋值给节点的 data 字段,并将 next 字段设置为 NULL。

c. 在链表末尾插入新节点

void insertAtEnd(Node** head, int data) {
    Node* newNode = createNode(data);
    if (newNode == NULL) {
        printf("内存分配失败!\n");
        return;
    }

    if (*head == NULL) {
        *head = newNode;
    } else {
        Node* temp = *head;
        while (temp->next != NULL) {
            temp = temp->next;
        }
        temp->next = newNode;
    }
    printf("已在链表末尾插入节点:%d", data);
}
  • 调用 createNode 函数创建一个新节点;
  • 检查内存分配是否成功;
    • 如果成功,则根据链表是否为空来确定新节点的位置。
  • 若链表为空,则将新节点设置为头节点;
  • 否则,遍历链表找到最后一个节点,并将最后一个节点的 next 指针指向新节点。

d. 删除指定节点

void deleteNode(Node** head, int data) {
    if (*head == NULL) {
        printf("链表为空!\n");
        return;
    }

    Node* temp = *head;
    Node* prev = NULL;

    if (temp != NULL && temp->data == data) {
        *head = temp->next;
        free(temp);
        printf("已删除节点:%d", data);
        return;
    }

    while (temp != NULL && temp->data != data) {
        prev = temp;
        temp = temp->next;
    }

    if (temp == NULL) {
        printf("节点 %d 不存在!\n", data);
        return;
    }

    prev->next = temp->next;
    free(temp);
    printf("已删除节点:%d", data);
}
  • 检查链表是否为空,如果为空则输出相应的提示信息。
  • 遍历链表,找到要删除的节点。
    • 如果找到了节点,则修改前一个节点的 next 指针,使其跳过要删除的节点,并释放该节点的内存空间。
    • 如果没有找到要删除的节点,则输出相应的提示信息。

e. 修改指定节点的数据

void modifyNode(Node* head, int oldData, int newData) {
    if (head == NULL) {
        printf("链表为空!\n");
        return;
    }

    Node* temp = head;
    while (temp != NULL) {
        if (temp->data == oldData) {
            temp->data = newData;
            printf("已将节点 %d 修改为 %d\n", oldData, newData);
            return;
        }
        temp = temp->next;
    }

    printf("节点 %d 不存在!\n", oldData);
}

查找~删除~修改……这里不重复介绍,懂的都懂,不懂我也没办法

f. 遍历链表并打印

void printList(Node* head) {
    if (head == NULL) {
        printf("链表为空!\n");
        return;
    }

    Node* temp = head;
    printf("链表节点数据:");
    while (temp != NULL) {
        printf("%d ", temp->data);
        temp = temp->next;
    }
    printf("\n");
}
  • 检查链表是否为空,如果为空则输出相应的提示信息。
  • 使用一个临时指针变量 temp 来遍历链表,依次访问每个节点并打印其数据。

g. 主函数

nt main() {
    Node* head = NULL;  // 头节点

    insertAtEnd(&head, 1);
    insertAtEnd(&head, 2);
    insertAtEnd(&head, 3);

    printList(head);

    deleteNode(&head, 2);

    printList(head);

    deleteNode(&head, 4);

    return 0;
}
  • 创建了一个头节点 head;
  • 调用 insertAtEnd 函数三次,在链表末尾插入了三个节点;
  • 调用 printList 函数打印链表的节点数据;
  • 调用 deleteNode 函数删除链表中的一个节点,并再次打印链表的节点数据;
  • 调用 deleteNode 函数尝试删除一个不存在的节点。

C语言代码整合

#include <stdio.h>
#include <stdlib.h>

// 定义链表节点结构
typedef struct Node {
    int data;
    struct Node* next;
} Node;

// 创建新节点
Node* createNode(int data) {
    Node* newNode = (Node*)malloc(sizeof(Node));
    if (newNode != NULL) {
        newNode->data = data;
        newNode->next = NULL;
    }
    return newNode;
}

// 在链表末尾插入新节点
void insertAtEnd(Node** head, int data) {
    Node* newNode = createNode(data);
    if (newNode == NULL) {
        printf("内存分配失败!\n");
        return;
    }

    if (*head == NULL) {
        *head = newNode;
    } else {
        Node* temp = *head;
        while (temp->next != NULL) {
            temp = temp->next;
        }
        temp->next = newNode;
    }
    printf("已在链表末尾插入节点:%d", data);
}

// 删除指定节点
void deleteNode(Node** head, int data) {
    if (*head == NULL) {
        printf("链表为空!\n");
        return;
    }

    Node* temp = *head;
    Node* prev = NULL;

    if (temp != NULL && temp->data == data) {
        *head = temp->next;
        free(temp);
        printf("已删除节点:%d", data);
        return;
    }

    while (temp != NULL && temp->data != data) {
        prev = temp;
        temp = temp->next;
    }

    if (temp == NULL) {
        printf("节点 %d 不存在!\n", data);
        return;
    }

    prev->next = temp->next;
    free(temp);
    printf("已删除节点:%d", data);
}
// 修改指定节点的数据
void modifyNode(Node* head, int oldData, int newData) {
    if (head == NULL) {
        printf("链表为空!\n");
        return;
    }

    Node* temp = head;
    while (temp != NULL) {
        if (temp->data == oldData) {
            temp->data = newData;
            printf("已将节点 %d 修改为 %d\n", oldData, newData);
            return;
        }
        temp = temp->next;
    }

    printf("节点 %d 不存在!\n", oldData);
}
// 遍历链表并打印节点数据
void printList(Node* head) {
    if (head == NULL) {
        printf("链表为空!\n");
        return;
    }

    Node* temp = head;
    printf("链表节点数据:");
    while (temp != NULL) {
        printf("%d ", temp->data);
        temp = temp->next;
    }
    printf("\n");
}

// 主函数测试链表操作
int main() {
    Node* head = NULL;  // 头节点

    insertAtEnd(&head, 1);
    insertAtEnd(&head, 2);
    insertAtEnd(&head, 3);

    printList(head);

    deleteNode(&head, 2);

    printList(head);

    deleteNode(&head, 4);

    return 0;
}

Cpp代码整合

        与C语言基本相同,这里不再过多介绍

#include <iostream>

// 定义链表节点结构
class Node {
public:
    int data;
    Node* next;

    // 构造函数
    Node(int data) : data(data), next(nullptr) {}
};

// 链表类
class LinkedList {
private:
    Node* head;

public:
    // 构造函数
    LinkedList() : head(nullptr) {}

    // 析构函数,用于释放链表内存
    ~LinkedList() {
        Node* current = head;
        while (current != nullptr) {
            Node* next = current->next;
            delete current;
            current = next;
        }
    }

    // 在链表末尾插入新节点
    void insertAtEnd(int data) {
        Node* newNode = new Node(data);

        if (head == nullptr) {
            head = newNode;
        } else {
            Node* temp = head;
            while (temp->next != nullptr) {
                temp = temp->next;
            }
            temp->next = newNode;
        }
        std::cout << "已在链表末尾插入节点:" << data << std::endl;
    }

    // 删除指定节点
    void deleteNode(int data) {
        if (head == nullptr) {
            std::cout << "链表为空!" << std::endl;
            return;
        }

        Node* temp = head;
        Node* prev = nullptr;

        if (temp != nullptr && temp->data == data) {
            head = temp->next;
            delete temp;
            std::cout << "已删除节点:" << data << std::endl;
            return;
        }

        while (temp != nullptr && temp->data != data) {
            prev = temp;
            temp = temp->next;
        }

        if (temp == nullptr) {
            std::cout << "节点 " << data << " 不存在!" << std::endl;
            return;
        }

        prev->next = temp->next;
        delete temp;
        std::cout << "已删除节点:" << data << std::endl;
    }

    // 修改指定节点的数据
    void modifyNode(int oldData, int newData) {
        if (head == nullptr) {
            std::cout << "链表为空!" << std::endl;
            return;
        }

        Node* temp = head;
        while (temp != nullptr) {
            if (temp->data == oldData) {
                temp->data = newData;
                std::cout << "已将节点 " << oldData << " 修改为 " << newData << std::endl;
                return;
            }
            temp = temp->next;
        }

        std::cout << "节点 " << oldData << " 不存在!" << std::endl;
    }

    // 遍历链表并打印节点数据
    void printList() {
        if (head == nullptr) {
            std::cout << "链表为空!" << std::endl;
            return;
        }

        Node* temp = head;
        std::cout << "链表节点数据:";
        while (temp != nullptr) {
            std::cout << temp->data << " ";
            temp = temp->next;
        }
        std::cout << std::endl;
    }
};

int main() {
    LinkedList list;

    list.insertAtEnd(1);
    list.insertAtEnd(2);
    list.insertAtEnd(3);

    list.printList();

    list.deleteNode(2);

    list.printList();

    list.deleteNode(4);

    return 0;
}

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

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

相关文章

循环日程安排问题(分治法)

函数接口定义&#xff1a; void Plan(int a[][N],int k); 裁判测试程序样例&#xff1a; #include #include <math.h> #include using namespace std; #define N 100 void Plan(int a[][N],int k); int main() { int i,j,a[N][N],k,size; cin>>k; sizepow(2,k)…

“滑动窗口”算法专项训练

目录 题目链接&#xff1a;长度最小的子数组 题目描述 思路分析&#xff1a;滑动窗口(利用单调性&#xff0c;使用"同向双指针来优化) 细节处理 画图解析 代码 题目链接&#xff1a;最大连续1的个数 III 题目描述 思路分析&#xff1a;滑动窗口(同向双指针) 细节…

面对纷繁复杂的低代码和无代码产品,开发者该如何选择?

凭借着革命性的生产力优势&#xff0c;低代码技术火爆了整个IT圈。面对纷繁复杂的低代码和无代码产品&#xff0c;开发者该如何选择&#xff1f; 在研究低代码平台的年数上&#xff0c;本人已有3年&#xff0c;也算是个低代码资深用户了&#xff0c;很多企业面临低代码选型上的…

TCP/IP(二十)TCP 实战抓包分析(四)TCP 第二次握手 SYN、ACK 丢包

一 实验二&#xff1a;TCP 第二次握手 SYN、ACK 丢包 重点&#xff1a; 通过设置 tcp_synack_retries 和 tcp_syn_retries内核参数,观察丢包的现象 ① 实验环境 iptables -t filter -I INPUT -s 172.25.2.100 -j DROPtcpdump -nni ens3 tcp and host 172.25.2.100 and por…

【算法 | 位运算No.1】leetcode268. 丢失的数字

个人主页&#xff1a;兜里有颗棉花糖 欢迎 点赞&#x1f44d; 收藏✨ 留言✉ 加关注&#x1f493;本文由 兜里有颗棉花糖 原创 收录于专栏【手撕算法系列专栏】【Leetcode】 &#x1f354;本专栏旨在提高自己算法能力的同时&#xff0c;记录一下自己的学习过程&#xff0c;希望…

打造一个开箱即用的超级丝滑的漂亮hexo博客网站

打造一个开箱即用的超级丝滑的漂亮hexo博客网站 目录 文章目录 打造一个开箱即用的超级丝滑的漂亮hexo博客网站目录文章持续更新地址鸣谢hexo简介hexo优缺点优点缺点&#x1f44e; 没有后台管理 博客效果主题特性版本迭代1.原版版本迭代2.自己版本迭代 贡献者主题下载&#x1f…

Python中if __name__ == ‘__main__‘

1. 在Python中&#xff0c;凡是以两个下划线开头&#xff0c;两个下划线结尾的变量叫做“魔法变量”。瓦特&#xff1f;魔法变量&#xff1f;对&#xff0c;你没有听错&#xff0c;就是魔法变量。所谓魔法变量就是Python对象内置天生就有的属性变量&#xff0c;你使用这些变量前…

开关电源测试方案介绍:如何进行电源耐压测试?

耐压测试是检验电源模块、电器设备等承受过压能力的测试方法&#xff0c;同时电源模块耐压测试还可以检测出设备的绝缘性能。在电气设备的使用过程中会出现电压突然上升的情况&#xff0c;有时也会因为天气原因出现高压&#xff0c;如果耐压性能弱&#xff0c;设备绝缘能力差&a…

【Java学习之道】SQL语言的基本语法与操作

引言 对于初学者来说&#xff0c;数据库编程可能听起来有些复杂&#xff0c;但实际上&#xff0c;只要掌握了SQL语言的基本语法和操作&#xff0c;你就能够轻松地处理各种数据问题。本章将为你揭示SQL语言的魅力&#xff0c;帮助你快速入门数据库编程。 一、SQL语言简介 SQL…

Three.js柏林噪音 流动球体

代码&#xff1a;https://gitee.com/honbingitee/three-template-next.js/tree/shader/ 参考油管视频&#xff1a;https://www.youtube.com/watch?voKbCaj1J6EI 核心&#xff1a; 创建循环的图形 应用噪声 顶点按照法相偏移 CustomMaterial(): ShaderMaterial {const material…

轻松上手,制作电子期刊就这么简单

嗨&#xff0c;年轻的朋友们&#xff01;你是否想过用你的创意和热情来制作一本属于自己的电子期刊&#xff1f;现在&#xff0c;这个梦想已经触手可及&#xff01;只需要用到这款工具即可轻松上手&#xff0c;就能拥有自己的电子期刊 工具&#xff1a;FLBOOK在线制作电子杂志平…

工具类app变现难?工具类产品广告变现策略实用指南

工具类app面临着买量成本高&#xff0c;收益提升难 的困境&#xff0c;在不同的变现周期可以采用不同的变现策略。#APP广告变现# 1、合理挖掘变现场景&#xff0c;提升eCPM 工具类 App 可基于自身产品属性和用户使用习惯路径&#xff0c;相应地选择开屏广告、信息流、横幅、…

品牌线上布局思路有哪些,品牌策略分析!

电商运营是现代企业宣推和销售产品的重要方式之一。要让品牌产品在电商节点&#xff0c;实现流量获取&#xff0c;就必须制定详细有效的品牌策略。今天为大家带来品牌线上布局思路有哪些&#xff0c;品牌策略分析&#xff01; 不同于线下销售广告&#xff0c;针对电商平台&…

浅谈压力测试的重要目标及意义

随着互联网应用的快速发展&#xff0c;软件系统的稳定性和性能成为了用户和企业关注的焦点。用户期望应用程序能够在高负载下依然保持稳定和高效。为了满足这一需求&#xff0c;压力测试成为了不可或缺的一环。本文将探讨压力测试的重要性以及如何进行压力测试。 一、压力测试的…

[正式学习java③]——字符串在内存中的存储方式、为什么字符串不可变、字符串的拼接原理,键盘录入的小细节。

一、字符串 1.字符串在内存中的存储方式 &#x1f525;在java中&#xff0c;内存中有两个地方可以存储字符串&#xff0c;一个是字符串池&#xff0c;一个是堆内存&#xff0c;串池中的字符串不会重复&#xff0c;而堆中的字符串每次都会开辟一块新的空间&#xff0c;因为维护…

Python武器库开发-基础篇(三)

基础篇(三) 函数 下面是一个打印问候语的简单函数&#xff0c;名为greet_user() &#xff1a; greeter.py def greet_user():"""显示简单的问候语"""print("Hello!")greet_user()这个示例演示了最简单的函数结构。第一行的代码行使…

优雅而高效的JavaScript——防抖和节流

&#x1f60a;博主&#xff1a;小猫娃来啦 &#x1f60a;文章核心&#xff1a;优雅而高效的JavaScript——防抖和节流 文章核心 引言事件处理的性能问题防抖&#xff1a;延迟触发事件基本原理实现防抖的代码示例实际应用场景举例 节流&#xff1a;限制事件触发频率基本原理实现…

传输层协议(TCP/UDP协议)

全文目录 端口号端口号范围划分 传输层UDP协议特点基于UDP的应用层协议 TCP协议确认应答机制&#xff08;可靠性&#xff09;延迟应答机制超时重传机制流量控制连接管理机制TIME_WAIT 状态CLOSE_WAIT 状态拥塞控制滑动窗口 TCP、UDP对比TCP的listen第二个参数 端口号 在套接字…

【算法设计zxd】第5章分治法

目录 分治算法策略的设计模式 分治思想&#xff1a; 分治算法求解问题的步骤&#xff1a; 设计模式 算法分析 二分查找算法 思考题 计算模型&#xff1a; 时间复杂度分析&#xff1a; 代码&#xff1a; 分治*大数乘法&#xff1a; 【例5-2】设X, Y是两个n位的十进制…

掌动智能分析云性能监控的重要性

云计算已成为现代企业的核心基础设施&#xff0c;它为企业提供了灵活性、可扩展性和成本效益。然而&#xff0c;将业务迁移到云上并不意味着问题就此消失。企业必须仍然保持对其云基础设施和应用程序的严格监控&#xff0c;以确保其性能和可靠性。本文将深入探讨企业云性能监控…