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

C语言链表详解附实例

什么是链表

链表是一种常见的重要的数据结构。它是动态地进行存储分配的一种结构。链表和数组比较,不用事先确定存储空间,而是根据需要开辟内存单元。
下图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;
	/*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<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语言链表详解内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • C语言单链表实现方法详解

    本文实例讲述了C语言单链表实现方法.分享给大家供大家参考,具体如下: slist.h #ifndef __SLIST_H__ #define __SLIST_H__ #include<cstdio> #include<malloc.h> #include<assert.h> typedef int ElemType; typedef struct Node { //定义单链表中的结点信息 ElemType data; //结点的数据域 struct Node *next

  • C语言中双向链表和双向循环链表详解

    双向链表和双向循环链表 和单向链表相比,多了一个前驱结点.如果他为空,那么next和prior都指向自己.而对于双循环链表,只需要最后一个元素的next指向head->next,head->next的prior指向最后一个节点即可. 插入操作 新节点s插入链表,s->next给p结点,s->prior给p->prior,然后,p->prior->next指向s,p->prior再指向s.顺序需要注意 s->next = p; s->prior =

  • C语言之复杂链表的复制详解

    什么是复杂链表? 复杂链表指的是一个链表有若干个结点,每个结点有一个数据域用于存放数据,还有两个指针域,其中一个指向下一个节点,还有一个随机指向当前复杂链表中的任意一个节点或者是一个空结点.今天我们要实现的就是对这样一个复杂链表复制产生一个新的复杂链表. 复杂链表的数据结构如下: typedef int DataType; //数据域的类型 //复杂链表的数据结构 typedef struct ComplexNode { DataType _data ; // 数据 struct Complex

  • C语言如何建立链表并实现增删查改详解

    前言 以下是本人完成的一个C语言建立链表并进行增删查改操作的程序,为方便学习,本人将整个程序分为头文件和主函数两部分: 1.头文件(函数部分) (1)初始化函数 #include <stdio.h> #include <stdlib.h> typedef struct { int *head; int length; int capacity; } Toslist; //Toslist类型 //初始化顺序表 Toslist initSeqlist() { Toslist list;

  • C语言之复杂链表的复制方法(图示详解)

    什么是复杂链表? 复杂链表指的是一个链表有若干个结点,每个结点有一个数据域用于存放数据,还有两个指针域,其中一个指向下一个节点,还有一个随机指向当前复杂链表中的任意一个节点或者是一个空结点.今天我们要实现的就是对这样一个复杂链表复制产生一个新的复杂链表. 复杂链表的数据结构如下: typedef int DataType; //数据域的类型 //复杂链表的数据结构 typedef struct ComplexNode { DataType _data ; // 数据 struct Complex

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

    C语言链表详解附实例 什么是链表 链表是一种常见的重要的数据结构.它是动态地进行存储分配的一种结构.链表和数组比较,不用事先确定存储空间,而是根据需要开辟内存单元. 下图1是最简单的一种链表(单向链表)的结构 第 0 个结点称为头结点,它存放有第一个结点的首地址,它没有数据,只是一个指针变量.以下的每个结点都分为两个域,一个是数据域,存放各种实际的数据,如学号 num,姓名 name,性别 sex 和成绩 score 等.另一个域为指针域,存放下一结点的首地址.链表中的每一个结点都是同一种结构类

  • C语言数据结构之单向链表详解分析

    链表的概念:链表是一种动态存储分布的数据结构,由若干个同一结构类型的结点依次串连而成. 链表分为单向链表和双向链表. 链表变量一般用指针head表示,用来存放链表首结点的地址. 每个结点由数据部分和下一个结点的地址部分组成,即每个结点都指向下一个结点.最后一个结点称为表尾,其下一个结点的地址部分的值为NULL(表示为空地址). 特别注意:链表中的各个结点在内存中是可以不连续存放的,具体存放位置由系统分配. 例如:int *ptr ; 因此不可以用ptr++的方式来寻找下一个结点. 使用链表的优点

  • C语言实现无头单链表详解

    目录 链表的结构体描述(节点) 再定义一个结构体(链表) 断言处理 & 判空处理 创建链表 创建节点 头插法 打印链表 尾插法 指定位置插入 头删法 尾删法 指定位置删除 查找链表 删除所有指定相同的元素 总结 再封装的方式,用 c++ 的思想做无头链表 链表的结构体描述(节点) #include <stdio.h> #include <stdlib.h> #include <assert.h> typedef int DataType; //节点 typede

  • Java逃逸分析详解及代码示例

    概念引入 我们都知道,Java 创建的对象都是被分配到堆内存上,但是事实并不是这么绝对,通过对Java对象分配的过程分析,可以知道有两个地方会导致Java中创建出来的对象并一定分别在所认为的堆上.这两个点分别是Java中的逃逸分析和TLAB(Thread Local Allocation Buffer)线程私有的缓存区. 基本概念介绍 逃逸分析,是一种可以有效减少Java程序中同步负载和内存堆分配压力的跨函数全局数据流分析算法.通过逃逸分析,Java Hotspot编译器能够分析出一个新的对象的

  • C语言链表与单链表详解

    链表是什么及链表的优势 链表是一种介于数组的另外一种数据结构: 我们知道数组可以存放很多的元素,这些元素都是呈线性排列,也就是一个挨着一个连续存放 但是当元素足够多时,还能继续正常的存放吗? 事实上的不可以的,虽然系统的内存足够大,但是这些内存不都是连续的,这就导致会出现没有足够的空间去存储这些元素. 其次就是数组的大小需要你去申请,如果你申请的空间足够大,就会导致内存的浪费 而链表就很好的解决了这两个问题 链表的组成 链表的作用就是相当与数组一样,储存你数据的 但又不同于数组,链表把每一个游离

  • 集合框架(Collections Framework)详解及代码示例

    简介 集合和数组的区别: 数组存储基础数据类型,且每一个数组都只能存储一种数据类型的数据,空间不可变. 集合存储对象,一个集合中可以存储多种类型的对象.空间可变. 严格地说,集合是存储对象的引用,每个对象都称为集合的元素.根据存储时数据结构的不同,分为几类集合.但对象不管存储到什么类型的集合中,既然集合能存储任何类型的对象,这些对象在存储时都必须向上转型为Object类型,也就是说,集合中的元素都是Object类型的对象. 既然是集合,无论分为几类,它都有集合的共性,也就是说虽然存储时数据结构不

  • C 语言基础----详解C中的运算符

    C语言中又有哪些运算符呢? 如下所示: ※ 算术运算符 ※ 赋值运算符 ※ 关系运算符 ※ 逻辑运算符 ※ 三目运算符 C语言基本算术运算符如下表: 除法运算中注意: 如果相除的两个数都是整数的话,则结果也为整数,小数部分省略,如果两数中有一个为小数,结果则为小数. 取余运算中注意: 该运算只适合用两个整数进行取余运算 运算后的符号取决于被模数的符号,如(-10)%3 = -1;而10%(-3) = 1. 注:C语言中没有乘方这个运算符,也不能用×,÷等算术符号. 赋值运算符 下表列出了 C 语

  • Java数据结构之链表详解

    一.链表的介绍 什么是链表 链表是一种物理存储单元上非连续.非顺序的存储结构,数据元素的逻辑顺序是通过链表中的指针链接次序实现的.链表由一系列结点(链表中每一个元素称为结点)组成,结点可以在运行时动态生成.每个结点包括两个部分:一个是存储数据元素的数据域,另一个是存储下一个结点地址的指针域. 相比于线性表顺序结构,操作复杂.由于不必须按顺序存储,链表在插入的时候可以达到O(1)的复杂度,比另一种线性表顺序表快得多,但是查找一个节点或者访问特定编号的节点则需要O(n)的时间,而线性表和顺序表相应的

  • C语言指针详解

    前言:复杂类型说明     要了解指针,多多少少会出现一些比较复杂的类型,所以我先介绍一下如何完全理解一个复杂类型,要理解复杂类型其实很简单,一个类型里会出现很多运算符,他们也像普通的表达式一样,有优先级,其优先级和运算优先级一样,所以我总结了一下其原则:从变量名处起,根据运算符优先级结合,一步一步分析.下面让我们先从简单的类型开始慢慢分析吧: int p; //这是一个普通的整型变量   int *p; //首先从P 处开始,先与*结合,所以说明P 是一个指针,然后再与int 结合,说明指针所

  • C语言 图文并茂详解程序编译过程

    目录 一.初识编译器 二.程序被编译的过程 三.小结 一.初识编译器 编译器是一个广义的概念,真正的编译器由下面几个模块组成,真正的编译器是进行语法分析和语义分析的. 二.程序被编译的过程 如下,file.i 是中间代码,file.s 是一个汇编文件,file.o 是二进制文件. 预编译 处理所有的注释,以空格代替 将所有的 #define 删除,并且展开所有的宏定义 处理条件编译指令 #if, #ifdef, #elif,#else,#endif 处理 #include,展开被包含的文件 保留

随机推荐