广告
返回顶部
首页 > 资讯 > 后端开发 > 其他教程 >C++深入分析讲解链表
  • 540
分享到

C++深入分析讲解链表

2024-04-02 19:04:59 540人浏览 泡泡鱼
摘要

目录链表的概述1、数组特点2、链表的概述3、链表的特点静态链表链表的操作1、链表插入节点头部之前插入节点尾部之后插入节点有序插入节点2、遍历链表节点3、查询指定节点4、删除指定节点5

链表的概述

1、数组特点

1、空间连续、元素类型相同、通过下标快速访问

2、静态数组:空间一旦确定不能更改(动态数组除外)

3、静态、动态数组 插入删除元素 需要移动大量数据

2、链表的概述

链表是一种物理存储上非连续,数据元素的逻辑连续通过链表节点中的指针变量保存下个节点的地址,实现的一种线性存储结构。

3、链表的特点

链表由一系列节点(链表中每一个元素称为节点)组成,节点在运行时动态生成(malloc,calloc),每个节点包 括两个部分:

数据域:存放节点数据(核心)

指针域:结构体指针变量 保存下一个节点的地址

静态链表

#include <stdio.h>
//设计节点类型
typedef struct stu
{
    //数据域
    int num;
    char name[32];
    float score;
    //指针域
    struct stu *next;
}STU;
int main(int arGC, char const *argv[])
{
    //静态链表的节点 不是从堆区申请
    STU node1={100, "lucy", 77.7f};
    STU node2={101, "bob", 66.7f};
    STU node3={102, "tom", 55.7f};
    STU node4={103, "hehe", 74.7f};
    STU node5={104, "xixi", 73.7f};
    //构建成链表
    STU *head = NULL;
    head = &node1;
    node1.next = &node2;
    node2.next = &node3;
    node3.next = &node4;
    node4.next = &node5;
    node5.next = NULL;
    //遍历链表(从头节点  逐个节点遍历)
    STU *pb = head;
    while(pb != NULL)
    {
        printf("%d %s %f\n", pb->num, pb->name, pb->score);
        //移动到下一个节点
        pb = pb->next;
    }
    return 0;
}

链表的操作

增、删、改、查

学生管理系统 讲解链表

1、链表插入节点

帮助信息函数:

void help(void)
{
    printf("******************************\n");
    printf("* insert:插入链表节点        *\n");
    printf("* printf:遍历链表节点        *\n");
    printf("* search:查询链表节点        *\n");
    printf("* delete:删除链表节点        *\n");
    printf("* free:释放链表节点          *\n");
    printf("* quit:遍历链表节点          *\n");
    printf("******************************\n"); 
    return; 
}

头部之前插入节点

STU* insert_link(STU *head, STU tmp)
{
    //为插入的节点申请空间
    STU *pi = (STU *)calloc(1,sizeof(STU));
    //给空间赋值
    *pi = tmp;
    pi->next = NULL;
    //判断链表是否存在
    if(NULL == head)//空
    {
        head = pi;
        //return head;
    }
    else//非空
    {
        pi->next = head;
        head = pi;
        //return head;
    }
    return head;
}

尾部之后插入节点

//尾部插入节点
STU* insert_link(STU *head, STU tmp)
{
    //为插入的节点申请空间
    STU *pi = (STU *)calloc(1,sizeof(STU));
    //给空间赋值
    *pi = tmp;
    pi->next = NULL;
    //判断链表是否存在
    if(NULL == head)
    {
        head = pi;
        return head;
    }
    else
    {
        //寻找尾部节点
        STU *pb = head;
        while(pb->next != NULL)
            pb = pb->next;
        //将pi插入到 尾节点后
        pb->next = pi;
        return head;
    }
    return head;
}

有序插入节点

//有序插入节点
STU* insert_link(STU *head, STU tmp)
{
    //为插入的节点申请空间
    STU *pi = (STU *)calloc(1,sizeof(STU));
    //给空间赋值
    *pi = tmp;
    pi->next = NULL;
    //判断链表是否存在
    if(NULL == head)
    {
        head = pi;
        return head;
    }
    else
    {
        //寻找插入点
        STU *pf = NULL, *pb = NULL;
        pf = pb = head;
        //从小--->大排序
        while((pb->num < pi->num) && (pb->next != NULL))
        {
            pf = pb;
            pb = pb->next;
        }
        if(pb->num >= pi->num)//头部插入、中部插入
        {
            if(pb == head)//头部插入
            {
                pi->next = head;
                head = pi;
            }
            else//中部插入
            {
                pf->next = pi;
                pi->next = pb;
            }
        }
        else if(pb->next == NULL)//尾部插入
        {
            pb->next = pi;
        }
    }
    return head;
}

2、遍历链表节点

void printf_link(STU *head)
{
    //1、判断链表是否存在
    if(NULL == head)
    {
        printf("链表不存在\n");
        return;
    }
    else//2、链表存在,逐个节点遍历链表
    {
        STU *pb = head;
        while(pb != NULL)
        {
            printf("%d %s %f\n", pb->num, pb->name, pb->score);
            //pb指向下一个节点
            pb = pb->next;
        }
    }
    return;
}

3、查询指定节点

STU *search_link(STU *head, char *name)
{
    //判断链表是否存在
    if(NULL == head)
    {
        printf("链表不存在\n");
        return NULL;
    }
    else//链表存在
    {
        //逐个节点查询
        STU *pb = head;
        while((strcmp(pb->name, name) != 0) && (pb->next != NULL))
        {
            pb = pb->next;
        }
        if(strcmp(pb->name, name) == 0)//找到
            return pb;
        else
        {
            printf("未找到相关节点信息\n");
            return NULL;
        }
    }
    return NULL;
}

4、删除指定节点

STU* delete_link(STU *head, char *name)
 {
     //判断链表是否存在
    if(NULL == head)
    {
        printf("链表不存在\n");
        return head;
    }
    else
    {
        STU *pf =NULL, *pb=NULL;
        pf = pb = head;
        //寻找删除点
        while((strcmp(pb->name, name) != 0) && (pb->next != NULL))
        {
            pf = pb;
            pb = pb->next;
        }
        //找到删除点
        if(strcmp(pb->name, name) == 0)
        {
            if(head == pb)//头部删除
            {
                head= head->next;
            }
            else//中尾部删除
            {
                pf->next = pb->next;
            }
            //释放pb
            if(pb != NULL)
            {
                free(pb);
                pb=NULL;
            }
        }
        else//未找到删除点
        {
            printf("未找到删除的相关节点信息\n");
        }
    }
    return head;
 }

5、释放链表

STU* free_link(STU *head)
 {
    //判断链表是否存在
    if(NULL == head)
    {
        printf("链表不存在\n");
        return head;
    }
    else//链表存在
    {
        STU *pb = head;
        while(pb != NULL)
        {
            //head纪录下一个节点的位置
            head = head->next;
            //释放pb指向的节点
            if(pb != NULL)
            {
                free(pb);
                pb = NULL;
            }
            //pb指向下一个节点
            pb=head;
        }
        printf("链表节点已经完全释放\n");
    }
    return head;
 }

6、链表的翻转

STU* reverse_link(STU *head)
 {
    //判断链表是否存在
    if(NULL == head)
    {
        printf("链表不存在\n");
        return head;
    }
    else
    {
        STU *pb = head->next;
        STU *pn = NULL;
        //将头结点指针域指向NULL
        head->next = NULL;
        //逐个节点翻转
        while(pb != NULL)
        {
            //pn纪录pb->next的节点
            pn = pb->next;
            //进行反向连接
            pb->next = head;
            //移动head到pb上
            head = pb;
            //pb指向pb的节点
            pb = pn;
        }  
    }
    return head;
 }

7、链表的排序

//选择法对链表排序
 void sort_link(STU *head)
 {
    //判断链表是否存在
    if(NULL == head)
    {
        printf("链表不存在\n");
        return;
    }
    else
    {
        STU *p_i = head, *p_j = head;//int i=0, j=0;
        while(p_i->next != NULL)//for(i=0;i<n-1; i++)
        {
            STU *p_min = p_i;//int min = i;
            p_j = p_min->next;//j=min+1
            while(p_j != NULL)//j<n
            {
                if(p_min->num > p_j->num)//if(arr[min] > arr[j])
                    p_min = p_j;//min = j;
                p_j = p_j->next;//j++
            }
            if(p_i != p_min)//if(i != min)
            {
                //整体交换
                STU tmp = *p_i;
                *p_i = *p_min;
                *p_min = tmp;
                //指针域交换
                tmp.next = p_i->next;
                p_i->next = p_min->next;
                p_min->next = tmp.next;
            }
            p_i = p_i->next;//i++
        }
    }
    return;
 }

到此这篇关于c++深入分析讲解链表的文章就介绍到这了,更多相关C++链表内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: C++深入分析讲解链表

本文链接: https://www.lsjlt.com/news/150524.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • C++深入分析讲解链表
    目录链表的概述1、数组特点2、链表的概述3、链表的特点静态链表链表的操作1、链表插入节点头部之前插入节点尾部之后插入节点有序插入节点2、遍历链表节点3、查询指定节点4、删除指定节点5...
    99+
    2022-11-13
  • C语言深入讲解链表的使用
    目录一、链表的概念二、链表的分类1. 单向或者双向链表2. 带头或者不带头(是否有自带哨兵位头结点)3. 循环或者非循环链表4. 无头单向非循环链表和带头双向循环链表3、链表的实现(...
    99+
    2022-11-13
  • C++深入分析讲解智能指针
    目录1.简介2.unique_ptr指针(独占指针)3.shared_ptr指针(共享所有权)4.weak_ptr(辅助作用)5.自实现初级版智能指针6.总结1.简介 程序运行时存在...
    99+
    2022-11-13
  • C++深入分析讲解类的知识点
    目录知识点引入类的初识1、封装2、权限3、类的定义(定义类型)4、类的成员函数与类中声明及类外定义Person类的设计设计立方体类点Point和圆Circle的关系知识点引入 C语言...
    99+
    2022-11-13
  • C++链式二叉树深入分析
    目录二叉树的结构和概念二叉树的操作前序遍历中序遍历和后序遍历二叉树的节点个数求二叉树叶子结点个数求二叉树的深度在二叉树查找为X的结点之前我们的重点学习二叉树都是完全二叉树,接下来我们...
    99+
    2022-11-13
  • JavaHashMap源码深入分析讲解
    1.HashMap是数组+链表(红黑树)的数据结构。 数组用来存放HashMap的Key,链表、红黑树用来存放HashMap的value。 2.HashMap大小的确定: 1) Ha...
    99+
    2022-11-13
  • Golangsync.Map原理深入分析讲解
    目录GO语言内置的mapsync.Mapsync.Map原理分析sync.Map的结构查找新增和更新删除GO语言内置的map go语言内置一个map数据结构,使用起来非常方便,但是它...
    99+
    2022-12-17
    Go sync.Map Golang sync.Map原理
  • C++深入分析讲解函数与重载知识点
    目录函数的默认(缺省)参数1、默认参数的定义2、默认参数的注意点占位参数1、占位参数 函数内部无法使用2、占位参数 可以设置成缺省参数函数重载函数的默认(缺省)参数 1、默认参数的定...
    99+
    2022-11-13
  • Java深入分析讲解反射机制
    目录反射的概述获取Class对象的三种方式通过反射机制获取类的属性通过反射机制访问Java对象的属性反射机制与属性配置文件的配合使用资源绑定器配合使用样例通过反射机制获取类中方法通过...
    99+
    2022-11-13
  • Spring深入分析讲解BeanUtils的实现
    目录背景DOBODTOVO数据实体转换使用方式原理&源码分析属性赋值类型擦除总结背景 DO DO是Data Object的简写,叫做数据实体,既然是数据实体,那么也就是和存储...
    99+
    2022-11-13
  • ReactHooks核心原理深入分析讲解
    目录Hooks闭包开始动手实现将useState应用到组件中过期闭包模块模式实现useEffect支持多个HooksCustom Hooks重新理解Hooks规则React Hook...
    99+
    2022-12-17
    React Hooks React Hooks原理
  • Python魔术方法深入分析讲解
    目录前言__init____new____call____del____str__总结前言 魔术方法就是一个类/对象中的方法,和普通方法唯一的不同是:普通方法需要调用,而魔术方法是在...
    99+
    2023-02-08
    Python魔术方法 Python魔术方法原理
  • Android权限机制深入分析讲解
    目录1、权限2、在程序运行时申请权限1、权限 普通权限:不会直接威胁到用户安全和隐私的权限危险权限:那些可能会触及用户隐私或者对设备安全性造成影响的权限。 到Android 10 系...
    99+
    2022-12-08
    Android权限机制 Android权限管理 Kotlin权限机制
  • C++深入讲解初始化列表的用法
    目录一、小问题二、类成员的初始化三、类中的 const 成员四、初始化与赋值的不同五、小结一、小问题 下面的类定义是否合法 如果合法,ci 的值是什么,存储在哪里 下面编写代码一探...
    99+
    2022-11-13
  • Java深入分析链表面试实例题目
    目录链表面试题一问题描述:问题分析:问题讲解:代码实现:链表面试题二问题描述:问题分析:问题讲解:代码实现:链表面试题一 判断链表是否是回文结构。 问题描述: 兄弟们,看图理解什么是...
    99+
    2022-11-13
  • C/C++深入讲解内存管理
    目录C/C++内存分布C语言中的动态内存管理C++的内存管理operator new与operator delete函数operator new与operator dele...
    99+
    2022-11-13
  • 深入浅析Java中的链表
    本篇文章为大家展示了深入浅析Java中的链表,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。单链表:insertFirst:在表头插入一个新的链接点,时间复杂度为O(1)deleteFirst:删除表...
    99+
    2023-05-31
    java ava 链表
  • AndroidView的事件分发机制深入分析讲解
    目录1.分发对象-MotionEvent2.如何传递事件1.传递流程2.事件分发的源码解析1.Activity对点击事件的分发过程2.顶级View对点击事件的分发过程3.主要方法4....
    99+
    2023-01-29
    Android View事件分发机制 Android事件分发
  • C++深入讲解哈夫曼树
    目录哈夫曼树的基本概念1)路径2)路径长度3)权4)结点的带权路径长度5)树的带权路径长度6)哈夫曼树哈夫曼树的构造算法哈夫曼树的构造过程哈夫曼树算法的实现1)结点的存储结构2)构建...
    99+
    2022-11-13
  • C++深入讲解函数重载
    目录函数重载概念重载依据值型别判断函数重载的规则名字粉碎-名字修饰函数重载 概念 在C++中可以为两个或者两个以上函数提供相同的函数名称,只要参数类型不同,或者参数数目不同,参数顺序...
    99+
    2022-11-13
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作