单链表的建立(头插法、尾插法)(数据结构与算法)

在这里插入图片描述

如果要把很多个数据元素存到一个单链表中,如何操作?
1.初始化一个单链表
2. 每次取一个数据元素,插入到表尾/表头

1. 尾插法建立单链表

尾插法建立的单链表元素顺序与输入数据集合的顺序相同,即按照输入数据的顺序排列。

使用尾插法建立单链表的一个常见应用是在计算机科学中进行数据输入。通过将用户输入的数据逐个添加到链表的尾部,可以方便地保存输入的数据,并在后续处理中使用。

  1. 初始化单链表
  2. 设置变量length纪录链表长度
  3. while循环{
  4. 每取一个元素e;
  5. ListInsert(L, length+1, e) 插入到尾部;
  6. length++;}
    在这里插入图片描述

尾插法(尾部插入法)是一种建立单链表的常用方法,与头插法相反,它会将新节点插入到链表的尾部位置。以下是使用尾插法建立单链表的步骤:

  1. 首先定义一个头节点,并将其初始化为空指针。
  2. 遍历需要转化为链表的数据集合。
  3. 对于数据集合中的每一个元素,创建一个新的节点,并设置其数据域为该元素的值。
  4. 若链表为空,将新节点直接设置为头节点。
  5. 若链表非空,则遍历到链表的最后一个节点,并将其指针域指向新节点。
  6. 更新链表的最后一个节点为新节点。
typedef struct LNode      //定义单链表结点类型
{ElemType data;		  //每个结点存放一个数据元素struct LNode *next;   //指针指向下一个节点
}
//初始化一个单链表(带头结点)
bool InitList(LinkList &L)  
{L = (LNode *)  malloc(sizeof(LNode)); //分配一个头结点if (L == NULL) 		 //内存不足, 分配失败return false;L->next = NULL; 		//头结点之后暂时还没有节点return true;	
}
void test()
{LinkList L;   //声明一个指向单链表的指针//初始化一个空表InitList(L);//......后续代码......
}//在第i个位置处插入元素e(带头结点)
bool ListInsert(LinkList &L, int i, ElemType e)
{if(i<1)return false;LNode *p;		//指针p指向当前扫描到的节点int j = 0; 		//指针p指向的是第几个结点p = L;		//L指向头结点 头结点是第0个结点,不存数据while( p != NULL && j < i-1) //循环找到第i-1个结点{p = p->next;j++;}if( p==NULL)	//i值不合法return false;LNode *s = (LNode *) malloc(sizeof(LNode));s->data = e;s->next = p->next;p->next = s;		//将结点s连到p之后return true;		//插入成功
}

当插入1个元素时,while需要循环一次,插入2个元素,while循环1此…插入n个元素,while循环n-1次。 每次都从开头开始之后遍历,循环次数为1+2+3+…+(n-1)。时间复杂度为0(n^2)

这种操作,时间复杂度太大,并不是最佳方案。

其实根本没必要每次都从头开始往后寻找, 我们可以设置一个指针r,指向表尾,当要在尾部插入一个新的数据元素时,就只需要对r结点做一个后插操作就行了。

//后插操作: 在p结点之后插入元素e
bool InsertNode(LNode *p, ElemType e)
{if(p == NULL)return false;LNode *s = (LNode *)malloc(sizeof(LNode));if(s==NULL)		//内存分配失败return false;s->data = e;		//用结点s保存数据元素s->next = p->next;   p->next = s; 		//将结点s连到p之后    后插操作return true;				
}

在这里插入图片描述

插入数据元素40

在这里插入图片描述

尾部插入一个元素,表尾指针后移一位,仍然指向最后一个元素,方便下一次插入

在这里插入图片描述

示例:尾插法建立单链表

设此次输入的整数是10,while循环检测,循环中前两句,会申请一个新的结点,然后让s这个指针指向新结点,并且把输入的x = 10放入新结点中,接下来把 r结点的next指针指向s结点。最后,再让r指针指向s结点,接下来就可以输入下一个元素x,继续插入。

时间复杂度:O(n)

LinkList List_TailInsert(LinkList &L)
{int x;		//设ElemType为整型L = (LinkList)malloc(sizeof(LNode));	//建立头结点LNode *s, *r = L;	//r为表尾指针scanf("%d", &x);	//输入结点的值while(x != 9999)    //该数是随便取的 ,输入9999表示结束{s = (LNode *) malloc(sizeof(LNode));  //简历一个结点ss->data = x; 	//将x存放到s数据域中r->next = s; //将表尾指针指向s结点r = s;	//r指向新的表尾结点scanf("%d",&x);}r->next = NULL; 	//尾结点指针置空return L;
}

LNode *s, *r = L; //r为表尾指针

在这里插入图片描述

申请一个新的结点,然后让s这个指针指向新结点,并且把输入的x = 10放入新结点中

在这里插入图片描述

接下来把 r 结点的next指针指向s结点

在这里插入图片描述

最后,再让r指针指向s结点, 接下来就可以输入下一个元素x,继续插入

在这里插入图片描述

假设继续输入 x = 16。那么会再申请一个新的结点s,让s指针指向新节点, 并把16放入新结点, 接下来把r结点的指针指向s结点, 最后…

在这里插入图片描述

将 r 指针指向s结点,再输入下一个元素27 。 以此类推完成插入操作。

在这里插入图片描述

如果x输入的是9999, 那么就可以跳过while循环,执行r->next = NULL; 让r结点的next指向NULL。
在这里插入图片描述

2. 头插法建立单链表

头插法建立单链表的特点是:新节点插入到链表的头部位置,因此建立完成的链表元素顺序是和输入数据集合的顺序相反的,即倒序排列。

对头结点的后插操作,如下图所示:

  1. 首先定义一个头节点,并将其初始化为空指针。
  2. 遍历需要转化为链表的数据集合。
  3. 对于数据集合中的每一个元素,创建一个新的节点,并设置其数据域为该元素的值。
  4. 将新节点的指针域指向当前头节点,即将新节点插入到链表的头部。
  5. 更新头节点为新节点。

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

//后插操作:在p结点之后插入元素e
bool InsertNextNode(LNode *p, ElemType e)
{if(p == NULL)return false;LNode *s = (LNode *)malloc(sizeof(LNode));if(s == NULL)	//内存分配失败return false;	s->data = e;		//用结点s保存数据元素es->next = p->next;p->next = s;		//将结点s连接到p之后return true;
}

在这里插入图片描述

  • 如上图所示,此处,尾插法中是没有 L->next = NULL 这一句的,但在头插法中,这句非常必要。由于是动态分配,如果头结点指向没有被初始化为NULL,那头结点L->next 很有可能指向了内存中某一神秘区域,从而形成脏数据,影响后面的插入操作。

养成好习惯,只要是初始化单链表,都先把头指针指向NULL

  • 按照头插法形成的规则,最终形成的单链表刚好是输入元素的逆序,这种性质非常重要!!!
    很多题目中都用得到单链表的逆置。
    在这里插入图片描述

3. 知识点回顾

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/184446.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

03运算符综合

03 3.1.1算数运算符 3.1.2赋值运算符 3.1.3比较&#xff08;关系&#xff09;运算符 3.1.4逻辑运算符 3.1.5位运算符 3.2运算符的优先级 3.3条件表达式

web3 React dapp项目通过事件从区块链中拿到 已取消 已完成 和所有的订单数据 并存入redux中

好 上文web3通过antd 在React dapp中构建订单组件基本结构我们算是把一个基本的订单组件展示做出来了 然后 我们继续 起一下环境先 ganache 终端运行 ganache -dMetaMask 登录一下 然后 打开项目 发布一下合约 truffle migrate --reset然后 运行一下 测试脚本 转入交易所 E…

linux地址空间

地址空间 内存空间示意图虚拟地址空间虚拟地址进程地址空间生命周期图解为什么要有地址空间呢&#xff1f; 小结 内存空间示意图 进程是在内存中运行的&#xff0c;为了便于管理&#xff0c;不同的数据会存储在不同的区域&#xff0c;因此内存就被分为几部分&#xff0c;如下图…

在MacBook上实现免费的PDF文件编辑

之前我想对PDF文件进行简单处理&#xff08;比如删页面、添空白页、调整页面顺序&#xff09;&#xff0c;要么是开wps会员【花钱贵】&#xff0c;下载&#xff08;盗版&#xff09;Adobe Acrobat【macOS不好下载】&#xff0c;要么用福昕阅览器登陆学生账号&#xff08;学校买…

逆向-文心一言开发者控制台调试

一打开标准的无限debugger 往上一层可以发现是jsvmp&#xff0c;这样替换文件相对来说就不太好搞 根据测试如果卡在debugger就会跳转页面 但是放行debugger就可以正常使用 可以基本确定debugger前后存在计时程序 这个时候就可以考虑对apply做hook劫持无限debugger的函数&#…

Eolink Apikit 版本更新:「数据字典」功能上线、支持 MongoDB 数据库操作、金融行业私有化协议、GitLab 生成 API 文档...

&#x1f389; 新增 搭建自定义接口协议架构&#xff0c;支持快速适配金融行业各类型私有协议的导入、编辑和展示。 数据字典功能上线&#xff0c;支持以数据字典的形式管理参数枚举值&#xff1b; 数据库连接支持 MongoDB 数据库操作&#xff1b; 基于 Apikit 类型导入 API…

Mac下flutter工程配置Gitlab cicd打包(暂时仅限android侧)

写的太粗糙&#xff0c;可能不太适合完全不懂的同学&#xff0c;但是实在没时间&#xff0c;而且也不太会写&#xff0c;权当做一个记录吧&#xff0c;对了还没有搞docker这些&#xff0c;还在持续学习中 1.GitLab Runner&#xff08;打包机&#xff09; 注意:需要有对应的权…

BIM、建筑机器人、隧道工程施工关键技术

一、BIM简介 &#xff08;一&#xff09;BIM概念 BIM&#xff08;Building Information Modeling&#xff09;&#xff0c;建筑信息模型。该技术通过数字化手段&#xff0c;在计算机中建立虚拟建筑&#xff0c;该虚拟建筑提供从单一到完整、包含逻辑关系的建筑信息库。信息库…

遇到不可复现的bug要怎么做?

测试人员遇到不可复现的bug要怎么做&#xff1f; 这是一个很常见的问题&#xff0c;也是一个很棘手的问题。不可复现的bug可能会给测试人员带来很大的困扰和压力&#xff0c;因为它们可能会影响软件的质量和用户的体验&#xff0c;但又很难找到问题的根源和解决方法。因此&…

软件测试/测试开发丨如何利用ChatGPT自动生成测试用例思维导图

点此获取更多相关资料 简介 思维导图是一种用图形方式表示思维和概念之间关系的工具&#xff1a; 有些公司会使用思维导图编写测试用例&#xff0c;这样做的优点是&#xff1a; 1.可视化和结构化。 2.易于理解&#xff0c;提高效率。 而 ChatGPT 是无法直接生成 xmind 格式…

深度学习中的数据类型介绍:FP32, FP16, TF32, BF16, Int16, Int8 ...

文章目录 0. 前言1. 数据的存储方式2. 不同数据类型介绍2.1 深度学习中常用的数据类型2.2 BF16 类型的优势2.3 不同数据类型的使用场景 0. 前言 相比于 CPU&#xff0c;GPU 在架构设计时将更多的晶体管用于数据处理&#xff0c;而不是数据缓存和流量控制&#xff0c;因此可以高…

Ansible 自动化运维工具 --- playbook 剧本

文章目录 1. Host inventory ---- 主机清单1.1 简介1.2 inventory文件1.3 Inventory 文件的构成1.3.1 主机与组1.3.2 变量 1.4 inventory 中的常用变量 2. Ansible-playbook剧本2.1 简介2.2 Playbook的结构组成2.3 编写playbook的基本格式与写法2.3.1 基本格式2.3.2 语句的横向…

【Linux】服务器与磁盘补充知识,硬raid操作指南

服务器硬件 cpu 主板 内存 硬盘 网卡 电源 raid卡 风扇 远程管理卡 1.硬盘尺寸: 目前生产环境中主流的两种类型硬盘 3.5寸 和2.5寸硬盘 2.5寸硬盘可以通过使用硬盘托架后适用于3.5寸硬盘的服务器 但是3.5寸没法转换成2.5寸 2.如何在服务器上制作raid 华为服务器为例子做…

Python中通过socketserver库创建服务端

socketserver库是Python的标准库&#xff0c;提供了套接字服务端的框架&#xff0c;通过该框架可以简化服务端的创建流程。 1 socketserver库的导入 通过如图1显示的代码导入socketserver库。 图1 导入socketserver库 2 通过socketserver库创建TCP服务端 通过socketserver库…

二维码智慧门牌管理系统升级解决方案:轻松实现辖区范围门址统计

文章目录 前言一、系统功能与优势 前言 在这个数字化时代&#xff0c;传统的门牌管理系统已经无法满足现代管理的需求。为了满足辖区内门址的统计需求&#xff0c;我们引入了全新的二维码智慧门牌管理系统升级解决方案。这一升级将让您轻松实现辖区范围门址的统计&#xff0c;…

20.7 OpenSSL 套接字SSL加密传输

OpenSSL 中的 SSL 加密是通过 SSL/TLS 协议来实现的。SSL/TLS 是一种安全通信协议&#xff0c;可以保障通信双方之间的通信安全性和数据完整性。在 SSL/TLS 协议中&#xff0c;加密算法是其中最核心的组成部分之一&#xff0c;SSL可以使用各类加密算法进行密钥协商&#xff0c;…

【嵌入式开发工具】STM32+Keil实现软件工程搭建与开发调试

本篇文章介绍了使用Keil来对STM32F103C8芯片进行初始工程搭建&#xff0c;以及开发与工程调试的完整过程&#xff0c;帮助读者能够在实战中体会到Keil这个开发环境的使用方法&#xff0c;了解一个嵌入式工程从无到有的过程&#xff0c;并且具备快速搭建一个全新芯片对应最小软件…

Ubuntu网络IP地址一直显示127.0.0.1

问题描述&#xff1a; 终端输入ip a显示127.0.0.1&#xff0c;原来类似192.168.231.1的地址不见了。 ip a 点击网络配置&#xff08;ubuntu桌面版&#xff09;&#xff0c;发现无线网络模块看不见了 正常情况应该有wired 模块&#xff0c;就是下面标红的 解决方案&#xff1a…

【有源码】基于uniapp的农场管理小程序springboot基于微信小程序的农场检测系统(源码 调试 lw 开题报告ppt)

&#x1f495;&#x1f495;作者&#xff1a;计算机源码社 &#x1f495;&#x1f495;个人简介&#xff1a;本人七年开发经验&#xff0c;擅长Java、Python、PHP、.NET、微信小程序、爬虫、大数据等&#xff0c;大家有这一块的问题可以一起交流&#xff01; &#x1f495;&…

Flutter的专属Skia引擎解析+用法原理

Skia是一款跨平台的2D图形库&#xff0c;是Google公司开发的&#xff0c;可以用于开发各种应用程序&#xff0c;如浏览器、游戏、移动应用程序等。Skia引擎的主要特点是速度快、可移植性强、占用的内存少、稳定性佳&#xff0c;适用于多种硬件平台。 Skia的目标是提供快速、高…