服务器之家:专注于VPS、云服务器配置技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - C/C++ - C语言数据结构与算法之单链表

C语言数据结构与算法之单链表

2022-07-24 18:07玄澈_ C/C++

单链表是一种链式存取的数据结构,用一组地址任意的存储单元存放线性表中的数据元素。本文将为大家介绍C语言中单链表的基本概念与读取数据元素,需要的可以参考一下

 

基本概念

C语言数据结构与算法之单链表

链表的每一个结点中只包含一个指针域

优点 : 储存空间利用高效

C语言数据结构与算法之单链表

举例来说:

typedef struct student{
	int id;		//学生编号
	char* name; //学生名称
 
	//指向下一结点的指针
	struct Student* pNext;
}Student;

与之相反的是多链表

C语言数据结构与算法之单链表

typedef struct student{
	int id;		//学生编号
	char* name; //学生名称
 
	//指向下一结点的指针
	struct Student* pNext;
	struct Student* qNext;
}Student;

 

读取数据元素

C语言数据结构与算法之单链表

 

获取第i个结点的数据元素

  1. 声明一个结点指针p指向链表的第一个结点a1,初始化j从1开始
  2. 当j < i 时,遍历链表,让p的指针向后移动,不断指向下一个结点,j 累加 1
  3. 当链表末尾 p 为空时,则说明第 i 个元素不存在;否则查找成功,返回结点 p 的数据

C语言数据结构与算法之单链表

1.定义数据元素

//定义数据元素
typedef struct student{
	int id;		
	char* name; 
}ElementType;

2.定义顺序表结构

typedef struct {
	ElementType dates[MAX_SIZE];   //当前顺序表中的数据集合
	int length;					   //当前顺序表中的元素个数
 
}SeqList;

3.定义链表的结点(包括数据域和指针域)

typedef struct Node {
	ElementType date;  //数据域
	struct Node* node; //指针域,指向下一个结点
}Node;

4.设置头结点

我们在定义链表时,习惯性的会定义头结点,以便统一链表结点的插入和删除操作

typedef struct Linklist {
	Node* next;  //头指针
	int length;
}Linklist;

如果链表有头结点,next就指向头结点,没有就指向第一个结点

链表的长度初始值为0

 

插入数据元素 

在第i个结点后插入数据元素  

  1. 创建一个空节点,分配内存空间,设置数据元素
  2. 获取第i个结点,设置新结点的后继结点为该结点的后继结点
  3. 设置第i个结点的后继结点为该结点

1.创建空节点并为数据域赋值

	//创建空节点并为数据赋值
	Node* node = (Node*)malloc(sizeof(Node));
    node -> date = element;
    node -> next = NULL;

2.通过循环找到要插入的结点

	for (int i = 1; currNode && i < pos - 1; i++)
	{
		currNode = currNode->next;
	}

3.将结点插入并对接前面的结点

	if (currNode) {
		node->next = currNode->next;
		currNode->next = node;
		linkList->length++;
	}

C语言数据结构与算法之单链表

 

初始化链表

void InitLinkList(LinkList* linkList, ElementType* dateArrar, int length)
{
	for (int i = 0; i < length; i++) {
		InsertLinkList(linkList, i + 1, dateArrar[i]);
	}
}

 

打印链表

void PrintLinkList(LinkList* linklist)
{
	Node* node = linklist->next;
	if (!node)
	{
		printf("链表为空!
");
		linklist->length = 0;
		return 0;
	}
	for (int i = 0; i < linklist->length; i++) {
		printf("%d	%s	
", node->date.id, node->date.name);
        node = node->next;
	}
}

 

顺序表查空

int IsLinkListEmpty(LinkList* linkList) {
 
	return linkList->length == 0 ? TRUE : FALSE;
 
}

 

顺序表的删除 

 

删除第i个结点及其数据元素

  1. 获取第i个结点,若该结点不是第一个结点,则获取第i - 1个结点
  2. 将第i -1个结点的后缀结点设为第i个结点的后缀结点
  3. 删除第i个结点,释放内存空间,记录并返回删除数据元素的值

C语言数据结构与算法之单链表

 

情况1:当删除的是第一个元素

	if (pos == 1)
	{
		node = linkList->next;
		if (node) {
			element = node->date;
			linkList->next = node->next;
			free(node);  //释放被删除的结点
			linkList->length--;
		}
        return element;
	}

 

情况2:除第一个结点外

  1. 找到要删除的结点和他的前缀结点
  2. 要删除结点的next 赋值给前缀结点
  3. 释放要删除的结点
	Node* preNode; //前缀结点
	node = linkList->next;
	for (int i = 1; node && i < pos; i++)
	{
		preNode = node;
		node = node->next;
	}
	if (node)
	{
		element = node->date;
		preNode->next = node->next;
		free(node);
		linkList->length--;
	}
	return element;

 

完整代码

ElementType DeleteLinkListElement(LinkList* linkList, int pos)
{
	ElementType element;   //被删除的元素
	element.id = -999;     //赋一个不可能的值,来判断删除是否成功
	Node* node = NULL;
 
	if (pos == 1)
	{
		node = linkList->next;
		if (node) {
			element = node->date;
			linkList->next = node->next;
			free(node);  //释放被删除的结点
			linkList->length--;
		}
	}
 
	Node* preNode; //前缀结点
	node = linkList->next;
	for (int i = 1; node && i < pos; i++)
	{
		preNode = node;
		node = node->next;
	}
	if (node)
	{
		element = node->date;
		preNode->next = node->next;
		free(node);
		linkList->length--;
	}
	return element;
 
}

 

删除单链表整表

  1. 声明结点p 和 q
  2. 将第一个结点赋值给p
  3. 循环将下一个结点赋值给q,释放p,将q赋值给p

C语言数据结构与算法之单链表

C语言数据结构与算法之单链表

C语言数据结构与算法之单链表

void CleatLinkList(LinkList* linkList)
{
	Node* node = linkList->next;
	Node* nextNode;
	while (node) {
		nextNode = node->next;  //先记录当前结点的下一个结点,以便释放当前结点的内存
		free(node);
		node = nextNode;
	}
	linkList->next = NULL;
	linkList->length = 0;
}

 

单链表VS顺序表

 C语言数据结构与算法之单链表

 C语言数据结构与算法之单链表

以上就是C语言数据结构与算法之单链表的详细内容,更多关于C语言单链表的资料请关注服务器之家其它相关文章!

原文链接:https://blog.csdn.net/forever_bryant/article/details/122026779

延伸 · 阅读

精彩推荐
  • C/C++wxWidgets实现无标题栏窗口拖动效果

    wxWidgets实现无标题栏窗口拖动效果

    这篇文章主要为大家详细介绍了wxWidgets实现无标题栏窗口拖动效果,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...

    zfwjsjzy6642021-07-22
  • C/C++C中实现矩阵乘法的一种高效的方法

    C中实现矩阵乘法的一种高效的方法

    本篇文章介绍了,在C中实现矩阵乘法的一种高效的方法。需要的朋友参考下...

    C语言教程网2912020-11-23
  • C/C++k均值算法c++语言实现代码

    k均值算法c++语言实现代码

    k均值算法c++语言实现代码...

    C++教程网6662021-01-08
  • C/C++C语言高效编程的几招小技巧

    C语言高效编程的几招小技巧

    这篇文章主要介绍了C语言高效编程的几招小技巧,本文讲解了以空间换时间、用数学方法解决问题以及使用位操作等编辑技巧,并给出若干方法和代码实例...

    C语言教程网10902021-02-26
  • C/C++C++实现LeetCode(72.编辑距离)

    C++实现LeetCode(72.编辑距离)

    这篇文章主要介绍了C++实现LeetCode(72.编辑距离),本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下...

    Grandyang3852021-11-30
  • C/C++使用WindowsAPI获取录音音频的方法

    使用WindowsAPI获取录音音频的方法

    这篇文章主要介绍了使用WindowsAPI获取录音音频的方法,非常实用的功能,需要的朋友可以参考下...

    C语言程序设计10822021-01-28
  • C/C++C++中4种类型转换方式 cast操作详解

    C++中4种类型转换方式 cast操作详解

    static_cast,支持子类指针到父类指针的转换,并根据实际情况调整指针的值,反过来也支持,但会给出编译警告,它作用最类似C风格的“强制转换”,一般...

    C++教程网11372021-01-05
  • C/C++C++选择文件夹代码的封装

    C++选择文件夹代码的封装

    这篇文章主要介绍了C++选择文件夹代码的封装,实例展示了将选择文件夹功能代码封装为一个类并对其进行实例化调用的过程,对于学习C++程序设计有不错的...

    C++教程网10642021-02-06