用c语言写链表

2023-05-16

        链表是数据结构的一种,是其他三个数据结构栈,树,图的基础,只有将链表这一数据结构弄懂,才能理解其他三种数据结构。

        举一个例子,老师让你设计一个联系人系统,其中包括姓名,电话号,住址,对于联系人系统中可以进行插入删除修改等基本操作,那么你将如何设计。

一.首先需要了解单链表

        单链表是链表这一数据的基础,单链表顾名思义就是将各种信息如姓名电话号住址像小盒子一样封装起来,再用链条将每个小盒子串到一块,只不过这个链条是单向的,只能从前面的盒子到后面的盒子,这样就形成了单链表。但我们学会单链表以后,其他的链表也就易如反掌了。

二.单链表的c语言定义

        从单链表的定义中来看,需要用struct结构体将单链表封装起来,结构体就相当于上面说的小盒子,包含有各种信息以及一个结构体指针。这个指针就相当于上面说的单向的链子,指向下一个小盒子,那么我们就可以很简单的实现定义。

         我们习惯叫这个小盒子为结点,单向的链子为指针。

        其中Node,为这个结构体的名字,LinkList为这个结构体的头结点,即第一个盒子,也就是整个链表开始的地方。因为我们需要用一个结构体指针来确保函数中对于一个链表的插入删除等操作可以成功实现,毕竟如果没有指针的话,在函数中的操作结果是不会影响外界的。同时对于不同链表的操作不会弄混,因为我们给一个链表起的名字不一样。

三.单链表的初始化

        对于一个单链表的初始化来说非常简单,我们采用头指针的方法,利用malloc()函数申请出空间,将其next单向链指向NULL即完成了对于单链表头结点的初始化。头结点的作用是方便对链表的各种操作。

 四.输入链表的元素

1.头插法

        头插法顾名思义就是将新来的元素从头结点处插入到链表中,具体插入方法如下图。

        s为一个新盒子,先将新盒子的单向链指向头结点单向链指向的盒子,再将头结点单向链指向这个新盒子,以此类推,将全部元素以这样的方法放入链表中,就形成了一个完整的链表。具体代码如下图:

        其中L为头指针,s为分配出的新结点空间 ,让c的数据储存在s的空间中,将这个新空间与链表链接起来,即完成了对于链表的创建。

         

2.尾插法

        尾插法与头插法相似,作用都是将元素插入到链表中,不同之处在于尾插法是将新的元素从尾部插入到链表中,具体插入方法如图。

         如图所示,其中s为一个新盒子,直接让该链表的尾部与这个新盒子用单向链子链接起来,这样以此类推,将所有元素插入就完成了对于链表的创建。具体代码如下。

        如图,其中rear为Node*类型,其始终指向该链表的尾部,保证后续元素顺利的插入到链表中。

五.元素的插入与删除

1.元素的插入

        在链表的操作中,需要把某个元素插入到链表的第i个位置中去,这时候就需要进行元素的插入。与头插法相似,代码如下图。

        其中p为新的盒子用来储存插入进去的元素, malloc为申请空间的函数。其中while循环为寻找第i - 1个盒子的位置,将s指向第i - 1个盒子。利用头插法将新盒子p插入到链表中,实现对于元素的插入。

2.元素的删除

        元素的删除相较于元素的插入还是相对简单的,仅仅需要找到第i个盒子,将第i - 1个盒子的单向链链到i + 1个盒子,再将第i个盒子的空间释放即可。具体代码如下。

                其中while循环为寻找第i - 1个盒子,用s指向它,然后让p指向第i个盒子,将s的单向链指向p的下一个盒子,最后将p释放即可完成对于元素的删除。

六.元素的输出

       在实际调试过程中,需要一步步将链表上的元素打印出来,以检测是否代码正确。实际代码如下。

        其中s为指向头结点的指针,一步步让指针指向下一个盒子,输出其中的元素,直到下一个盒子为空,这样就完成了对于链表的输出。

七.单链表的拓展

1.循环链表

        循环链表是单链表的延申,也是让程序员更加便于运用链表所诞生出来的链表。循环链表顾名思义就是将该链表的最后一个结点的单向链子指向了头结点,从而形成了一个循环。循环链表的初始化如下。

s -> next = s;

             这样该链表的指针L就指向了最后一个盒子,其中头结点为L的单向链子指的那个盒子,了解完循环链表的远离之后。其他的各种算法就可以很方便的实现了。

2.双向链表 

        双向链表就是在单向链的基础上加一个反方向的单向链,相当于相邻两个盒子是相通的可以互相到达的。对于双向链表的代码定义如下。

         其中prior为指向前一个的盒子的结点,与next的操作相似,需要进行初始化。循环链表增加了对于前一个盒子的可访问性,这样就能很容易的完成单向链表的某些操作。

八.其他

        链表是四大数据结构的基础,栈、树、图都需要以链表为基础进行编写,所以学好单链表是非常重要的。

#include<stdio.h>
#include<stdlib.h>

/*@author Lifan
  @function 链表的实现*/ 

typedef struct Node{
	char data;            //各种信息 
	struct Node *next;    //指针指向下一个结点 
}Node, *LinkList;


/*-----------函数声明---------------*/
void InitList(LinkList &L);            //初始化链表 
void CreateFromHead(LinkList &L);      //头插法插入数据 
void PrintList(LinkList &L);           //打印数组 
void CreateFromRear(LinkList &L);      //尾插法插入数据 
Node *FindElement(LinkList &L, char c); //查找元素 
int FindListLength(LinkList &L);        //求链表长度 
void InsertListElem(LinkList &L, int i, char c); //把元素c插入到第i个位置上 
void DeleteListElem(LinkList &L, int i, char &c); //将第i个位置的元素删除,保存在c中
/*----------------------------------*/ 
int main()
{
	LinkList L;
	InitList(L);
	//CreateFromHead(L);       //头插法 
	CreateFromRear(L);       //尾插法 
	PrintList(L);
/*******元素c是否在链表中************/
//	Node *r;
//	r = FindElement(L, 'c');
//	if(r != NULL) printf("%c", r -> data);

/*******元素c是否在链表中************/
//	printf("length:%d", FindListLength(L));

/*********将元素插入**********/
//	InsertListElem(L, 5, 's');
//	PrintList(L);

/*********将元素删除**********/
//	char ch;
//	DeleteListElem(L, 5, ch);
//	PrintList(L);
//	printf("ch:%c", ch);

	
	return 0;
}

void InitList(LinkList &L)
{
	L = (LinkList)malloc(sizeof(Node));
	L -> next = NULL;
}

void CreateFromHead(LinkList &L)
{
	Node *s;
	char c;
	bool isEnd = true;
	while(isEnd)
	{
		c = getchar();
		if(c != '.')
		{
			s = (Node *)malloc(sizeof(Node));
			s -> next = L -> next;
			L -> next = s;
			s -> data = c;
		}
		else isEnd = false;
	}
}

void CreateFromRear(LinkList &L)
{
	Node *s, *rear;
	rear = L;
	char c;
	bool isEnd = true;
	while(isEnd)
	{
		c = getchar();
		if(c != '.')
		{
			s = (Node *)malloc(sizeof(Node));
			s -> data = c;
			rear -> next = s;
			rear = s;
		}
		else 
		{
			rear -> next = NULL;
			isEnd = false;
		}
	}
}

void PrintList(LinkList &L)
{
	Node *s;
	s = L;
	while(s -> next != NULL)
	{
		s = s -> next;
		printf("%c", s -> data);
	}
	printf("\n");
}

Node *FindElement(LinkList &L, char c)
{
	Node *s;
	int i = 0;
	s = L -> next;
	while(s != NULL)
	{
		if(s -> data != c) s = s -> next;
		else 
		{
			return s;
			break; 
		}
		i++;
	}
	return NULL;
}

int FindListLength(LinkList &L)
{
	int length;
	Node *s;
	s = L -> next;
	while(s != NULL)
	{
		length++;
		s = s -> next;
	}
	return length;
}

void InsertListElem(LinkList &L, int i, char c)
{
	Node *s, *p;
	int j = 1;
	s = L -> next;
	p = (Node *)malloc(sizeof(Node));
	if(p == NULL) 
		printf("空间请求错误!!!");
	else while(s != NULL && j < i - 1) 
	{
		j++;
		s = s -> next;
	}
	p -> next = s -> next;
	s -> next = p;
	p -> data = c; 
	
}

void DeleteListElem(LinkList &L, int i, char &c)
{
	Node *s, *p;
	int j = 1;
	s = L -> next;
	while(s != NULL && j < i - 1)
	{
		j++;
		s = s -> next;
	}
	p = s -> next;
	c = p -> data;
	s -> next = p -> next;
	free(p); 
}

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

用c语言写链表 的相关文章

  • STM32外设串口资源用完了怎么办--------串口模拟解决问题(再也不用多个STM32或其它MCU)

    之前做项目的时候遇到了一个问题 xff0c 当把MCU本身的串口资源用完的时候 xff0c 却还需要使用多几个串口 xff0c 又不想使用几个MCU解决这个问题 那么模拟串口是解决这个问题的一种方法 下图是我对串口通信时序图的个人理解 xf
  • scanf函数中的格式字符串及注意事项

    scanf函数称为格式输入函数 xff0c 即按用户指定的格式从键盘上把数据输入到指定的变量之中 scanf函数的一般形式为 xff1a scanf 格式控制字符串 地址表列 xff1b 格式字符串的一般形式为 xff1a 输入数据宽度 长
  • 【C/C++】标准库, STL, Boost等的联系

    Backto C C 43 43 Index 标准库 最最开始 只有 C 语言 使用着使用着 一些常用的功能被写成了库 各种组织都是自己私有的库 后来为了方便统一使用和交流 就制定了标准 标准里的库 就是 C 标准库 后来 C 43 43
  • 数组与链表的优缺点和区别

    1 数组 xff1a 数组是将元素在内存中连续存放 xff0c 由于每个元素占用内存相同 xff0c 可以通过下标迅速访问数组中任何元素 但是如果要 在数组中增加一个元素 xff0c 需要移动大量元素 xff0c 在内存中空出一 个元素的空
  • 堆空间与栈空间的区别

    1 栈区 xff08 stack xff09 xff1a 又编译器自动分配释放 xff0c 存放函数的参数值 xff0c 局部变量的值等 xff0c 其操作方式类似于数据结构的 栈 2 堆区 xff08 heap xff09 xff1a 一
  • strtok函数及其实现

    头文件 xff1a include lt string h gt 定义函数 xff1a char strtok char s const char delim 函数说明 xff1a strtok 用来将字符串分割成一个个片段 参数s 指向欲
  • 服务器和客户机的信息函数以及读写函数

    1 服务器和客户机的信息函数 xff08 1 xff09 字节转换函数 在网络上面有着许多类型的机器 xff0c 这些机器在表示数据的字节顺序是不同的 xff0c 比如i386芯片是低字节在内存地址的低 端 xff0c 高字节在高端 xff
  • TCP协议的拥塞控制机制

    最初的TCP协议只有基于窗口的流控制 xff08 flow control xff09 机制而没有拥塞控制机制 流控制作为接受方管理发送方发送 数据的方式 xff0c 用来防止接受方可用的数据缓存空间的溢出 流控制是一种局部控制机制 xff
  • 构造函数为什么不能是虚函数

    1 从存储空间角度 xff0c 虚函数对应一个指向vtable虚函数表的指针 xff0c 这大家都知道 xff0c 可是这个指向vtable的指针其实是存储在对象的内存空间的 问题出来了 xff0c 如果构造函数是虚的 xff0c 就需要通
  • openmv学习五:OLED

    首先需要将SSD1306x py这个文件放到OPENMV中 代码 from machine import I2C Pin 从 machine 模块导入 I2C Pin 子模块 from ssd1306x import SSD1306 I2C
  • 自旋锁的实现及优化

    自旋锁也是一种互斥锁 xff0c 和mutex锁相比 xff0c 它的特点是不会阻塞 xff0c 如果申请不到锁 xff0c 就会不断地循环检测锁变量的状态 xff0c 也就是自旋 自旋锁的实现算法大多使用TAS算法或者CAS算法 TAS算
  • C语言----详解字符串相关的库函数(建议收藏)

    文章目录 系列文章目录前言一 C语言相关字符串库函数一览表 二 strlen函数 三 strcpy函数四 strcat函数 五 strcmp函数 六 strncpy函数 七 strncat函数 八 strncmp函数 九 strstr函数
  • C/C++头文件与变量的声明和定义

    C C 43 43 头文件与变量的声明和定义 最近遇到了变量重复包含的问题 xff0c 才发现自己有好多知识已经模糊了 xff0c 真惭愧 首先说下头文件 xff0c 其实头文件对计算机而言没什么作用 xff0c 她只是在预编译时在 inc
  • C语言寄存器变量register

    用register声明的变量是寄存器变量 xff0c 是存放在CPU的寄存器里的 而我们平时声明的变量是存放在内存中的 虽说内存的速度已经很快了 xff0c 不过跟寄存器比起来还是差得远 寄存器变量和普通变量比起来速度上的差异很大 xff0
  • Curl(C++)使用教程

    1 Curl简介 2 Easy interface 3 Multi interface 1 Curl简介 libcurl作为是一个多协议的便于客户端使用的URL传输库 xff0c 基于C语言 xff0c 提供C语言的API接口 xff0c
  • GPS原始RMC数据解析之DDMM.MMMM

    环境 GPS BD 定位模块 1 模块输出数据如下 GNRMC 100756 000 V 4000 0005 N 11559 9745 E 6 21 223 00 050313 N 68 2 了解换算规则 ddmm mmmm规则和dd dd
  • VINS-Mono代码阅读笔记(三):vins_estimator中main函数分析

    在VINS Mono代码阅读笔记 xff08 一 xff09 xff1a 从Euroc数据集的运行开始 中我们已经初步知道了vins estimator中订阅和发布的topic xff0c 那么 xff0c 在研究vins estimato
  • VINS-Mono代码阅读笔记(十三):posegraph中四自由度位姿优化

    本篇笔记紧接着VINS Mono代码阅读笔记 xff08 十二 xff09 xff1a 将关键帧加入位姿图当中 xff0c 来学习pose graph当中的紧耦合优化部分 在重定位完成之后 xff0c 进行位姿图优化是为了将已经产生的所有位
  • VINS-Mono代码阅读笔记(十四):posegraph的存储和加载

    本篇笔记紧接着VINS Mono代码阅读笔记 xff08 十三 xff09 xff1a posegraph中四自由度位姿优化 xff0c 来分析位姿图的存储和加载 完整 xff08 也是理想的 xff09 的SLAM的使用应该是这样的 xf

随机推荐

  • Visual Studio中设置opencv环境

    图像处理的项目中 xff0c 每建立一个新的项目 xff0c 需要对环境重新设置 xff0c 本文记录一下自己在VS中设置环境的步骤 xff0c 也分享给相同的入门小白 本文侧重说明VS中调用opencv的环境设置步骤 xff0c open
  • STM32学习笔记(1)---工程创建

    文章目录 前言一 新建工程步骤1 点击Project gt New uVision Project2 选择芯片3 创建文件夹4 放入必要文件4 头文件关联 总结问题 前言 这是我第一次写博客 xff0c 而且对于STM32也只是初学 xff
  • 乌班图安装 Kalibr

    安装ROS Melodic 1 Installation 1 1 Configure your Ubuntu repositories http www 360doc com content 18 0417 15 54525756 7463
  • ubuntu16.04 安装 php7.0-curl

    sudo apt add repository ppa ondrej php 添加这个源 sudo apt get update sudo apt get install php7 0 curl 这时成功安装php7 0 curl
  • string、char *、char[] 相互转换转换

    点击打开原文链接 一 string 转 char 主要有三种方法可以将 str 转换为 char 类型 xff0c 分别是 xff1a data c str copy 1 data 方法 xff1a string str 61 34 hel
  • STM32F103标准库开发---Uart串口通信实验---函数发送和中断接收

    STM32F103标准库开发 目录 文章目录 一 Uart串口通信 函数发送 1 Uart串口发送 标准库 函数 单字节发送 2 Uart串口检测标志 标准库 函数 3 Uart串口函数发送具体程序 二 Uart串口通信 中断接收 1 Ua
  • Keil5----新建项目文件( .c文件 和 .h文件)

    前言 在使用 Keil5 编辑程序的时候 xff0c 一定需要新建几个文件 xff08 c文件 和 h文件 xff09 xff0c 在其中编写不同功能的程序 例如 xff1a 新建LED c和LED h文件 xff0c 实现LED灯闪烁的功
  • 编码和串口通信

    先了解字符串和bytes xff08 字节 xff09 字符串 xff1a python里的字符串就是文本 xff0c 用于与人类交互 xff0c 像这样 xff1a 阿拉伯数字 xff1a a 61 1234566454 英语 xff1a
  • vscode终端不显示,闪退问题解决(完整步骤)

    1 以管理员身份运行此程序 步骤 xff1a 1 1 找到该文件目录的文件图标 1 2 右键属性选择兼容性 1 3 选择更改所有用户的设置然后勾选以管理员身份运行此程序后重新打开vscode 2 在vscode修改配置文件 2 1 打开vs
  • Vue项目启动报错 error:cannot find module xxx

    原因 xff1a 无法找到项目依赖的某个模块 解决办法 xff1a 1 删掉存放模块的文件夹node module xff1b 2 执行清除缓存命令 npm cache clean xff1b 如果报错 xff0c 使用强制清除npm ca
  • OkHttp-(一)HttpUrl了解

    1 xff0c git地址 xff1a https github com square okhttp 2 xff0c 官网地址 xff1a https square github io okhttp Http作为现代应用程序的常用联网方式
  • 学习网络编程第一步,安装NetAssist网络调试助手

    x1f4d6 摘要 今天分享下 遇到 Request header is too large xff0c 如何解决 xff0c 欢迎关注 xff01 x1f91e 简单介绍 NetAssist 是一款免安装的网络调试助手工具 今天给大家带来
  • 初学STM32之串口通信

    文章目录 一 背景知识1 处理器与外部通信的两种方式2 串行通信的三种传输方式3 串行通信的通信方式 二 串口通信基础1 STM32的串口通信接口2 UART异步通信引脚连接方法3 UART异步通信方式特点4 串口异步通信需要定义的参数 三
  • 前端架构图解

  • Ubuntu 18.04快捷安装ROS Melodic及rosdep update time out的问题解决

    1 ROS快捷安装 以下安装指令汇总针对Ubuntu18 04的ROS Melodic版本 xff1a 强烈建议复制以下指令到新建的xxx sh文件中 xff0c 保存后给xxx sh权限 xff0c 然后执行脚本一路输入y等候安装完成 e
  • NVIDIA Jetson AGX Xavier学习笔记3——环境配置(pytorch、torchvision、cv2)

    最近研究中需要使用NVIDIA Jetson AGX Xavier人工智能开发组件 由于也是第一次接触相关硬件设备 xff0c 遇到了很多困难 在这里记录整个Jetson AGX Xavier组件的学习过程 其中很多内容网上有比较详细的教程
  • Linux网络编程——tcp实例

    题目 1 通过TCP协议实现多个client端可以并发连接到server xff0c client可获得server指定目录下的文件列表 span class hljs comment client c Created on 2016年11
  • A星寻路算法的学习总结(详解)

    目录 1 理论基础 1 1A星寻路是用来解决什么问题的 1 2A星寻路的基本原理 2 代码实现 2 1每个格子的信息 2 2A星寻路管理器 2 3测试代码 3 实例演示 1 理论基础 1 1A星寻路是用来解决什么问题的 A星寻路是用来计算玩
  • C语言单片机栈、堆、堆栈的区别(仅供参考)

    计算机C语言中各个变量的存放区域 xff1a 代码区 xff08 CODE xff09 xff1a 存放函数代码 xff1b 静态数据区 xff08 DATA xff09 xff1a 存放全局变量 静态变量 xff1b 堆区 xff08 H
  • 用c语言写链表

    链表是数据结构的一种 xff0c 是其他三个数据结构栈 xff0c 树 xff0c 图的基础 xff0c 只有将链表这一数据结构弄懂 xff0c 才能理解其他三种数据结构 举一个例子 xff0c 老师让你设计一个联系人系统 xff0c 其中