iis服务器助手广告广告
返回顶部
首页 > 资讯 > 后端开发 > 其他教程 >C语言链表详解及代码分析
  • 372
分享到

C语言链表详解及代码分析

2024-04-02 19:04:59 372人浏览 安东尼
摘要

目录什么是链表环境构建建立静态链表包含所需要的头文件宏定义相关变量创建一个结构体主函数结果展示说明建立动态链表包含所需要的头文件宏定义相关变量创建一个结构体建立链表函数主函数结果展示

什么是链表

链表是一种常见的重要的数据结构。它是动态地进行存储分配的一种结构。链表和数组比较,不用事先确定存储空间,而是根据需要开辟内存单元。
下图1是最简单的一种链表(单向链表)的结构

图一

第 0 个结点称为头结点,它存放有第一个结点的首地址,它没有数据,只是一个指针变量。以下的每个结点都分为两个域,一个是数据域,存放各种实际的数据,如学号 num,姓名 name,性别 sex 和成绩 score 等。另一个域为指针域,存放下一结点的首地址。链表中的每一个结点都是同一种结构类型。

环境构建

用的Visual Studio 2019软件

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

在源文件中添加C文件

在这里插入图片描述

建立静态链表

包含所需要的头文件


#include<stdio.h> //标准输入输出头文件
#include<stdlib.h>//包含了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;
	
	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<stdio.h> //标准输入输出头文件
#include<stdlib.h>//包含了C、C++语言的最常用的系统函数
#include<malloc.h>//动态存储分配函数头文件

宏定义相关变量


#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;
}

##链表的删除

删除链表的元素也就是把前节点的指针域越过要删除的节点指向下下个节点。即:p->next = q->next;然后放出q节点的空间,即free(q);

在这里插入图片描述

删除函数


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;
}

结果显示

在这里插入图片描述

出现的问题

1、出现scanf 和printf 在VS2019中使用时会出错,解决办法如下

在这里插入图片描述

在这里插入图片描述

最后是测试的所有源程序

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

--结束END--

本文标题: C语言链表详解及代码分析

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

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

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

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

下载Word文档
猜你喜欢
  • C语言链表详解及代码分析
    目录什么是链表环境构建建立静态链表包含所需要的头文件宏定义相关变量创建一个结构体主函数结果展示说明建立动态链表包含所需要的头文件宏定义相关变量创建一个结构体建立链表函数主函数结果展示...
    99+
    2022-11-12
  • C语言链表与单链表详解
    链表是什么及链表的优势 链表是一种介于数组的另外一种数据结构: 我们知道数组可以存放很多的元素,这些元素都是呈线性排列,也就是一个挨着一个连续存放 但是当元素足够多时,还能继续正常的...
    99+
    2022-11-13
  • C语言数据结构之单向链表详解分析
    链表的概念:链表是一种动态存储分布的数据结构,由若干个同一结构类型的结点依次串连而成。 链表分为单向链表和双向链表。 链表变量一般用指针head表示,用来存放链表首结点的地址。 每个...
    99+
    2022-11-12
  • 详解C语言之单链表
    目录一、思路步骤1. 定义结构体2.初始化3.求当前数据元素的个数4.插入5.删除6.释放内存空间二、代码总结 一、思路步骤 1. 定义结构体 a.数据域:用来存放数据 b.指针域...
    99+
    2022-11-12
  • C语言单链表实例分析
    今天小编给大家分享一下C语言单链表实例分析的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。1、移除链表元素链接直达:移除链表元...
    99+
    2023-06-30
  • C语言线性表的链式表示及实现详解
    目录前言代码实现1. 单链表的结点构造2. 构造一个空的头结点3. 对线性表进行赋值4.对线性表进行销毁5.对线性表进行重置6.判断线性表是否为空7.获取线性表的长度8.获取线性表某...
    99+
    2022-11-13
  • C语言线性表之双链表详解
    目录定义1.删除2.插入3.建立4.查找总结定义 链表是通过一组任意的存储单元来存储线性表中的数据元素,每一个结点包含两个域:存放数据元素信息的域称为数据域,存放其后继元素地址的域称...
    99+
    2022-11-13
  • C语言:代码宏详解
    目录1、定义宏2、宏函数3、多行宏4、宏变长参数5、原样输出变量名6、例子7、宏与函数的差异总结1、定义宏 #define ARRAY_SIZE 100 double data[...
    99+
    2022-11-12
  • C语言中单链表的示例分析
    这篇文章将为大家详细讲解有关C语言中单链表的示例分析,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。一、思路步骤1. 定义结构体a.数据域:用来存放数据b.指针域:用来存放下一个数据的位置2.初始化申请头结...
    99+
    2023-06-25
  • C语言类的双向链表详解
    目录前言双向链表的定义双向链表的创建节点的创建双向链表节点查找双向链表的插入双向链表的节点删除双向链表的删除总结前言 链表(linked list)是一种这样的数据结构,其中的各对象...
    99+
    2022-11-12
  • C语言 分支语句详解分析
    目录分支语句的分类1、单分支结构语法形式:2、双分支结构3、多分支结构注意事项分支语句的分类 在不同的场景,我们常常面临着选择,有时候会在爱情与面包之间选择,有时候会在尊严与生存之间...
    99+
    2022-11-12
  • C语言算法的定义及分析详解
    目录算法的定义算法和程序的区别算法程序算法的性质算法的表示算法的分析分析原则常用的复杂性函数算法分析基本法则非递归算法:总结算法的定义 算法是一系列良定义的计算步骤 算法和程序的区别...
    99+
    2022-11-12
  • C++实现双向链表代码分析
    目录前言:一、双向链表优缺点二、C++实现分析(1)节点类(2)链表类分析(3)链表类构造函数(4)isEmpty()判断是否为空(5)size()获取链表长度(6)getNode(...
    99+
    2022-11-13
  • c语言爱心代码详解
    以下是一个爱心形状的C语言代码:```c#include int main() {int i, j;int n = 6; // 控制...
    99+
    2023-09-28
    c语言
  • C语言实现无头单链表详解
    目录链表的结构体描述(节点)再定义一个结构体(链表) 断言处理 & 判空处理创建链表创建节点头插法打印链表尾插法 指定位置插入 头删法尾删法&n...
    99+
    2022-11-13
  • c语言单链表反转代码怎么写
    以下是一个简单的C语言单链表反转代码示例: #include #include // 定义链表节点结构体 typedef st...
    99+
    2023-10-26
    c语言
  • C语言详解strcmp函数的分析及实现
    目录1.函数介绍1.1.函数接口1.2.函数分析1.3.函数的简单使用1.4.函数使用结果分析2.库函数strcmp源代码2.1.库函数源代码2.2.库函数分析3.模拟实现 strc...
    99+
    2022-11-13
  • C语言数据结构之双链表&循环链表&静态链表详解
    目录单链表 VS 双链表双链表双链表的初始化(带头结点)双链表的插入双链表的删除双链表的遍历循环单链表循环双链表循环双链表的初始化循环双链表的插入循环双链表的删除静态链表什么是静态链...
    99+
    2022-11-13
  • C语言实现动态链表的示例代码
    目录结构体定义已经函数声明函数实现创建一个链表判断链表是否为空获得链表中节点的个数在某个特定的位置插入一个元素获得指定下标的节点的元素删除一个节点链表逆序链表的清空链表的销毁链表的遍...
    99+
    2022-11-13
  • C语言学习之链表的实现详解
    目录一、链表的概念二、链表的结构三、顺序表和链表的区别和联系四、链表的实现一、链表的概念 链表是一种物理存储结构上非连续、非顺序的存储结构,数据元素的逻辑顺序是通过链表中的指针链接次...
    99+
    2022-11-13
    C语言 链表实现 C语言 链表
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作