IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> 数据结构之单向链表 -> 正文阅读

[数据结构与算法]数据结构之单向链表

单向链表

  • 功能模块

1//初始化链表
    LinkList init_LinkLIst()
2//插入链表
void insert_LinkList(LinkList list, int pos, void* data)
3//遍历链表
void foreach_LinkList(LinkList list,void(*myForeach)(void*))
4//删除链表 ,按位置
void removeByPos_LinkList(LinkList list, int pos)
5//按值删除链表
void removeByValue_LinkList(LinkList list,void * data,int(*myCompare)(void*,void*))
6//清空链表
void clear_LinkList(LinkList list)
7//销毁链表
void destroy_LinkList(LinkList list)
8//返回链表长度
int size_LinkList(LinkList list)
  • 细节讲解

1、typedef void* LinkList; //这一步是为了防止用户直接操作数据
//在后面使用时,可以转成目标数据结构
	struct LList* myList = list;

2、在初始化函数中,要给相应的数据赋初值

3、在插入和删除节点中,不要忘记改变m_size的值

4、在删除节点中,是利用两个辅助指针,来进行链表重新连接

5、注意回调函数的使用
#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <math.h>
#include <time.h>
#include <Windows.h>


//节点结构体
struct LinkNode
{
	//数据域----->因为不知道用户数据类型,故用void *【泛型】
	void* data;
	//指针域
	struct LinkNode* next;

};

//链表结构体
struct LList
{
	//头节点
	struct LinkNode pHeader;
	//链表长度
	int m_size;

};

typedef void* LinkList; //这一步是为了防止用户直接操作数据
//在后面使用时,可以转成目标数据结构

//初始化链表
LinkList init_LinkLIst()
{
	struct LList* myList = malloc(sizeof(struct LList));
	if (myList == NULL)
	{
		return NULL;
	}

	myList->pHeader.data = NULL;
	myList->pHeader.next = NULL;
	myList->m_size = 0;

	return myList;
}

//插入链表
void insert_LinkList(LinkList list, int pos, void* data)
{
	if (list == NULL)
	{
		return;
	}
	if (data == NULL)
	{
		return;
	}

	//将list还原成struct LList数据类型
	struct LList* myList = list;
	if (pos < 0 || pos > myList->m_size)
	{
		//无效位置,--->强制做尾插
		pos = myList->m_size;
	}

	//找到插入节点的前驱节点位置
	struct LinkNode* pCurrent = &myList->pHeader;

	for (int i = 0; i < pos; i++)
	{
		pCurrent = pCurrent->next;
	}

	//pCurrent 要插入节点的前驱

	//创建新节点
	struct LinkNode* newNode = malloc(sizeof(struct LinkNode));
	newNode->next = NULL;
	newNode->data = data;

	//建立节点关系
	newNode->next = pCurrent->next;
	pCurrent->next = newNode;

	//更新链表长度
	myList->m_size++;

}

//遍历链表
void foreach_LinkList(LinkList list,void(*myForeach)(void*))
{
	if (list == NULL)
	{
		return;
	}

	struct LList* myList = list;

	struct LinkNode* pCurrent = myList->pHeader.next;

	for (int i = 0; i < myList->m_size; i++)
	{
		//这里因为不知道数据类型,故使用回调函数
		myForeach(pCurrent->data);
		pCurrent = pCurrent->next;
	}


}

//删除链表 ,按位置
void removeByPos_LinkList(LinkList list, int pos)
{
	if (list == NULL)
	{
		return;
	}

	struct LList* myList = list;
	if (pos < 0 || pos > myList->m_size-1 )
	{
		return;
	}

	//找到待删除节点的前驱节点
	struct LinkNode* pCurrent = &myList->pHeader;
	for (int i = 0; i < pos; i++)
	{
		pCurrent = pCurrent->next;
	}

	//记录待删除的节点
	struct LinkNode* pDel = pCurrent->next;

	//重新建立节点关系
	pCurrent->next = pDel->next;

	free(pDel);
	pDel = NULL;

	//更新链表长度
	myList->m_size--;
}

//按值删除链表
void removeByValue_LinkList(LinkList list,void * data,int(*myCompare)(void*,void*))
{
	if (list == NULL)
	{
		return;
	}
	if (data == NULL)
	{
		return;
	}

	struct LList* myList = list;

	//创建两个辅助指针
	struct LinkNode* pPrev = &myList->pHeader;
	struct LinkNode* pCurrent = myList->pHeader.next;

	for (int i = 0; i < myList->m_size; i++)
	{
		//pCurrent->data  data 将两个指针比较利用回调 交给用户
		if (myCompare(pCurrent->data,data))
		{
			pPrev->next = pCurrent->next;

			free(pCurrent);
			pCurrent = NULL;

			myList->m_size--;
			break;
		}

		//辅助指针后移
		pPrev = pCurrent;
		pCurrent = pCurrent->next;

	}
}

//清空链表
void clear_LinkList(LinkList list)
{
	if (list == NULL)
	{
		return;
	}

	struct LList* myList = list;

	struct LinkNode* pCrrent = myList->pHeader.next;

	for (int i = 0; i < myList->m_size; i++)
	{
		struct LinkNode* pNext = pCrrent->next;

		free(pCrrent);
		pCrrent = pNext;
	}

	myList->pHeader.next = NULL;
	myList->m_size = 0;



}

//返回链表长度
int size_LinkList(LinkList list)
{
	if (list == NULL)
	{
		return -1;
	}
	struct LList* myList = list;
	return myList->m_size;
}

//销毁链表
void destroy_LinkList(LinkList list)
{
	if (list == NULL)
	{
		return;
	}
	//清空链表

	clear_LinkList(list);

	free(list);
	list = NULL;
}



//测试

struct Person
{
	char name[64];

	int age;

};

void myPrintPerson(void* data)
{
	struct Person* p = data;
	printf("姓名:%s		年龄:%d\n", p->name, p->age);
}

void myComparePerson(void* data1, void* data2)
{
	struct Person* p1 = data1;
	struct Person* p2 = data2;

	return strcmp(p1->name, p2->name) && p1->age == p2->age;
}


void test01()
{
	//准备数据
	struct Person p1 = { "亚瑟", 18 };
	struct Person p2 = { "妲己", 20 };
	struct Person p3 = { "安琪拉", 19 };
	struct Person p4 = { "凯", 21 };
	struct Person p5 = { "孙悟空", 999 };
	struct Person p6 = { "李白", 999 };


	//初始化链表
	LinkList mylist = init_LinkLIst();
	
	printf("链表的长度为:%d\n", size_LinkList(mylist));

	//插入数据
	insert_LinkList(mylist, 0, &p1);
	insert_LinkList(mylist, 0, &p2);
	insert_LinkList(mylist, -1, &p3);
	insert_LinkList(mylist, 0, &p4);
	insert_LinkList(mylist, 1, &p5);
	insert_LinkList(mylist, 0, &p6);

	//遍历
	foreach_LinkList(mylist, myPrintPerson);
	printf("链表的长度为:%d\n", size_LinkList(mylist));

	//测试删除
	removeByPos_LinkList(mylist, 4);
	printf("------------------\n");

	foreach_LinkList(mylist, myPrintPerson);
	printf("链表的长度为:%d\n", size_LinkList(mylist));


	//按值删除

	// 李白 凯 孙悟空 妲己 亚瑟 安琪拉
	struct Person p = { "孙悟空",999 };
	removeByValue_LinkList(mylist, &p, myComparePerson);
	printf("------------------\n");

	foreach_LinkList(mylist, myPrintPerson);
	printf("链表的长度为:%d\n", size_LinkList(mylist));

	//测试清空
	clear_LinkList(mylist);

	//返回链表长度
	printf("链表长度为:%d\n", size_LinkList(mylist));

	//销毁
	destroy_LinkList(mylist);
	mylist = NULL;

}

int main(void)
{
	test01();
	system("pause");
	return EXIT_SUCCESS;
}

测试截图

在这里插入图片描述

  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2021-07-04 20:01:20  更:2021-07-04 20:01:55 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年4日历 -2024/4/17 4:55:15-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码