C语言链表详解附实例

C语言链表详解附实例链表是一种常见的重要的数据结构 它是动态地进行存储分配的一种结构 链表和数组比较 不用事先确定存储空间 而是根据需要开辟内存单元 本文章实现了用 C 语言编写学生管理系统 可以实现添加 删除 增加 等功能

C语言链表详解附实例

什么是链表

环境构建

在这里插入图片描述
在源文件中添加C文件
在这里插入图片描述




建立静态链表

包含所需要的头文件

#include 
  
    //标准输入输出头文件 #include 
   
     //包含了C、C++语言的最常用的系统函数 
    
  

宏定义相关变量

#define LEN sizeof(struct Student)//宏定义节点长度得命名 #define TYPE struct Student//宏定义结构体变量命名 

创建一个结构体

struct Student//定义一个学生类型结构体,包括学号,分数 { long num; float score; struct Student* next;//next是指针变量,指向结构体变量 }; //指向结构体对象得指针变量既可以指向结构体变量,也可以指向结构体数组中得元素 

主函数

int main() { TYPE* head,*p;//定义头指针 struct Student a,b,c;//定义三个结构体变量 a.num = 101; a.score = 20;//分别对三个结点赋值 b.num = 102; b.score = 20; c.num = 103; c.score = 20; /*1、A.B则A为对象或者结构体 2、A->B则A为指针,->是成员提取,A->B是提取A中的成员B,A只能是指向类、结构、联合的指针;*/ head = &a; a.next = &b; b.next = &c; c.next = NULL; p = head;//把首地址给变量 do { printf("%ld %5.1f\n",p->num,p->score);//输出每个结点信息 p = p->next;//使P指向下一个结点 } while (p != NULL);//直到指针域指向空值 return 0; } 

结果展示

在这里插入图片描述

说明

将第一个结点的起始地址赋值给头指针head,将第二个结点的起始地址赋值给第一个结点的next成员,将第二个结点的起始地址赋给第一个结点的next…第三个结点的next赋值为NULL,这就形成了简单的链表。

建立动态链表

所谓建立动态链表是指在程序执行过程中从无到有地建立起一个 链表,即一个一个地开辟结点和输入各结点数据,并建立起前后相连的关系。

包含所需要的头文件

#include 
  
    //标准输入输出头文件 #include 
   
     //包含了C、C++语言的最常用的系统函数 #include 
    
      //动态存储分配函数头文件 
     
    
  

宏定义相关变量

#define LEN sizeof(struct Student)//宏定义节点长度得命名 #define TYPE struct Student//宏定义结构体变量命名 

创建一个结构体

struct Student//定义一个学生类型结构体,包括学号,分数 { long num; float score; struct Student* next;//next是指针变量,指向结构体变量 }; //指向结构体对象得指针变量既可以指向结构体变量,也可以指向结构体数组中得元素 

建立链表函数

TYPE* Creat(void)//定义函数,此函数返回一个指向链表头的指针 { TYPE* head;//定义头指针 TYPE* p1,*p2;//定义两个 指针变量用来相互保存 number = 0;//开始时,结点清零 p1 = p2 = (TYPE*)malloc(LEN);//创建存储空间 printf("请按格式输入学生学号,分数\n");//输出提示信息 printf("例如101,1 并以0,0结束\n"); scanf("%ld,%f", &p1->num, &p1->score);//按格式输入第一个结点的信息 head = NULL;//第一个结点头指针赋空值 while (p1->num!=0)//循环直到输入学生学号为0,就结束 { number++;//结点自增 if (number == 1)//如果只有一个结点,那么头指针指向第一个输入的结点 head = p1; else p2->next = p1;//如果大于1个,那么要用next保存前一个结点的信息 p2 = p1;//保存前一个结点信息 p1 = (TYPE*)malloc(LEN);//开辟新的结点 scanf("%ld,%f", &p1->num, &p1->score);//输入下一个结点信息 } p2->next = NULL;//循环结束,将指向信息赋空值 return (head);//返回首地址 } 

主函数

int main() { TYPE* pt;//定义一个结构体指针变量 pt = Creat();//函数返回链表第一个结点的地址 printf("\nnum:%ld\nscore:%5.lf\n", pt->num,pt->score);//输出第一个结点的成员值 return 0; } 

结果展示

在这里插入图片描述
== 文中最后结果显示的是第一个结点的内容,作为有强大功能的链表,对他的操作当然有许多,比如:链表的创建,修改,删除,插入,输出,排序,反序,清空链表的元素,求链表的长度等等。==

链表的输出

用循环直接可以输出链表

输出函数

void print(TYPE * head) { TYPE * p;//定义指针 printf("\nNOW These %d records are:\n");//输出显示信息 p = head;//使p指向第一个结点 if(head!=NULL)//输出第一个结点后的信息 do { printf("%ld %5.1f\n",p->num,p->score); p = p->next;//指向下个结点 } while (p != NULL); } 

主函数

int main() { TYPE * pt;//定义一个结构体指针变量 pt = Creat();//函数返回链表第一个结点的地址 print(pt);//输出调用 return 0; } 

链表的修改

修改函数

修改链表节点值很简单。下面是一个传入链表和要修改的节点,来修改值的函数.

void change(TYPE* head, int n) //修改指定位置的结点的信息 { TYPE* p = head;//传入首地址 int i = 0; while (i < n && p != NULL) { p = p->next; i++; }//找到相应的位置结点 if (p != NULL) { printf("输入要修改的值\n"); scanf("%ld,%f", &p->num, &p->score);//输入下一个结点信息 } else printf("节点不存在\n"); } 

主函数

int main() { TYPE* pt;//定义一个结构体指针变量 pt = Creat();//函数返回链表第一个结点的地址 change(pt,2);//修改相关结点的信息,假设修改第2+1个 print(pt);//输出调用 return 0; } 

删除函数

void delet(TYPE* head, int n) { TYPE* p = head, * in;//定义两边指针 int i = 0; while (i < n && p != NULL) { in = p;//找到左边的 p = p->next;//找到右边的 i++; } if (p != NULL) { in->next = p->next;//将左右链接 free(p);//释放中间结点 } else { printf("节点不存在\n"); } } 

主函数

int main() { TYPE* pt;//定义一个结构体指针变量 pt = Creat();//函数返回链表第一个结点的地址 delet(pt,1);//删除第1+1个结点 print(pt);//输出调用 return 0; } 

输出结果

在这里插入图片描述
链表的插入
我们可以看出来,插入节点就是用插入前节点的指针域链接上插入节点的数据域,再把插入节点的指针域链接上插入后节点的数据域。根据图,插入节点也就是:e->next = head->next; head->next = e;
增加链表节点用到了两个结构体指针和一个int数据。


在这里插入图片描述






插入函数

void insert(TYPE* head, int n) {//链表的插入 TYPE* p = head, * in; int i = 0; while (i < n && p != NULL) { p = p->next; i++;//找到相应结点 } if (p != NULL) { in = (TYPE*)malloc(sizeof(TYPE));//开辟新的空间 printf("输入要插入的值\n"); scanf("%ld,%f", &in->num, &in->score);//输入新的结点信息 in->next = p->next;//填充in节点的指针域,也就是说把in的指针域指向p的下一个节点 p->next = in;//填充p节点的指针域,把p的指针域重新指向in } else { printf("节点不存在\n"); } } 

主函数

int main() { TYPE* pt;//定义一个结构体指针变量 pt = Creat();//函数返回链表第一个结点的地址 insert(pt, 1);//从1+1后插入 print(pt);//输出调用 return 0; } 

结果显示

在这里插入图片描述

出现的问题

最后是测试的所有源程序

链接: https://download.csdn.net/download/xiaoxiaodawei/.

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/209238.html原文链接:https://javaforall.net

(0)
上一篇 2026年3月19日 上午9:43
下一篇 2026年3月19日 上午9:43


相关推荐

  • 使用FSO修改文件特定内容的函数

    使用FSO修改文件特定内容的函数function FSOchange(filename,Target,String)Dim objFSO,objCountFile,FiletempDataSet objFSO = Server.CreateObject(“Scripting.FileSystemObject”)Set objCountFile = objFSO.OpenTextFile(Server.MapPath(fil

    2022年5月31日
    36
  • mac怎么装linux双系统_双系统linux和windows

    mac怎么装linux双系统_双系统linux和windows第一步:格式化U盘第二步:下载系统,这里我选择的是基于manjaro第三步:将iso镜像转成dmg格式第四步:写入镜像第五步:分空间第六步:关闭OSX的-SIP保护第七步:安装refind第八步:重启按住option键安装系统第九步:重启查看结果第一步:格式化U盘第二步:下载系统,这里我选择的是基于manjaro第三步:…

    2026年4月17日
    2
  • java 新建项目_java怎么新建项目?java新建项目实操案例

    java 新建项目_java怎么新建项目?java新建项目实操案例java新建项目是学习java最基础的实操了,最近有小伙伴想知道java怎么新建项目?那么下面我们就来给大家讲解一下java新建项目的方法。1、选择“file(文件)”|“new(新建)”|“JavaProject(Java项目)”命令,打开“NewJavaProject(新建Java项目)”对话框。2、设置“Projectname(项目名)”为HelloJava,选中“Usedefau…

    2022年7月7日
    33
  • pytorch lstm时间序列预测问题踩坑「建议收藏」

    这里写目录标题1.做时间序列问题2.问题1.数据集自己做,为多个输入对应多个或一个输出2.损失函数注意:不能用交叉熵nn.CrossEntropyLoss()3.准确率1.做时间序列问题2.问题1.数据集自己做,为多个输入对应多个或一个输出2.损失函数注意:不能用交叉熵nn.CrossEntropyLoss()nn.CrossEntropyLoss()要求target目标值即真实值是标签,是torch.int64类型数据,即整数,不允许小数,如果输入小数会强行取整,应该用nn.MSELo

    2022年4月16日
    45
  • altium designer怎么画圆弧_word怎么画小圆形

    altium designer怎么画圆弧_word怎么画小圆形1.概述近期在学PADS这个平台,禁止覆铜区域可以放置各种形状,而在AD一直用的是polygonpourcutout这个选项,似乎不能绘制出圆形呢?别着急,我们接着往下看,AD还是上手比较快的设计软件。2.步骤如上所述,一般我们在AD要禁止覆铜,或者挖去铜皮,一般都用polygonpourcutout:但涉及如下图(蓝色圆圈)这样的禁止覆铜,就不好操作了。…

    2026年3月7日
    5
  • js动画requestAnimationFrame详解「建议收藏」

    js动画requestAnimationFrame详解「建议收藏」看这篇文章之前我希望你会用setTimeout做简单的动画,也就是利用递归来代替setInterval做动画。requestAnimationFrame()他的作用就是代替定时器做更加流畅高性能的动画,做可以匹配设备刷新率的动画,他解决了定时器做动画时间间隔不稳定的问题(也就是解决定时器做动画不流畅的问题)。他的用法与setTimeout差不多。与setTimeout一样的是都会返回一个唯一标识,setTimeout可以通过clearTImeout()关闭定时器。那么requestAnimatio.

    2022年8月31日
    7

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号