数据结构和算法学习记录——线性表之单链表(上)-初始单链表及其尾插函数(顺序表缺陷、单链表优点、链表打印)

简介: 数据结构和算法学习记录——线性表之单链表(上)-初始单链表及其尾插函数(顺序表缺陷、单链表优点、链表打印)

单链表的概念

单链表是一种链式存取的数据结构,链表中的数据是以结点来表示的。

每个结点的构成:元素(数据元素的映象) + 指针(指示后继元素存储位置)。

元素就是存储数据的存储单元,指针就是连接每个结点的地址数据。

以“结点的序列”表示的线性表称作线性链表(单链表),单链表是链式存取的结构。


复习一下之前学过的顺序表:

顺序表的缺陷

1.当空间不够时需要增容,而增容需要付出代价,浪费一些空间。

2.为了避免频繁增容,一般地,增容时都增至原本容量的2倍;这样就会可能存在一部分空间没有被使用的情况。

3.顺序表的数据是从开始的位置连续存储的,当需要从头部或者中间插入或删除数据时,就需要挪动数据,效率不高。


顺序表的优势

1.支持随机访问。

(有些算法需要结构支持随机访问,比如:二分查找、优化的快速排序等等)

而链表呢

链表的优势.1.按需申请空间,不用了就释放空间(更合理地使用空间)。 2.在头部或中间插入或删除数据时,不需要挪动数据,效率较高。


链表的缺陷

1.每一个数据,都要存一个指针去链接后面的数据结点。

2.不支持随机访问。

单链表的结构

逻辑结构

想象出来的、便于理解的

物理结构

内存中实际如何存储的

代码呈现

//单链表的基本结构
#define SLTDataType int
typedef struct SListNode
{
    SLTDataType data;
    struct SListNode * next;
}SLTNode;

单链表的打印

//单链表的打印
void SListPrint(SLTNode* phead)
{  //创建一个临时结点,将phead赋给cur
  SLTNode* cur = phead;
  
  //判断cur是否等于空
   while(cur != NULL)
   {
     //打印链表数据
     printf("%d->",cur->data);
     
     //找下一块的数据,就要找下一个结点
     //而下一个结点存放于cur中,通过cur找到下一个结点
     //一直循环,直到next指针指向NULL时,链表打印完成
     cur = cur->next;
   }
}

单链表的尾插函数

要实现单链表尾插,首先判断传入的链表是否为空,如果为空,则直接给其赋值。

如果不为空,则需要找到最后一个结点,即尾结点。尾结点的特征是:next指针指向NULL。

通过这一特征找到尾结点,再将新结点赋给尾结点。同时新结点的next指针指向NULL。

//单链表尾插函数
void SListPushBack(SLTNode ** pphead,SLTDataType x)
{  //创建一个新结点
   SLTNode * newnode = (SLTNode *)malloc(sizeof(SLTNode));
   newnode->data = x;
   newnode->next = NULL;
  //当传入的链表为空时,要先给其赋值,才能进行后面的操作
  //二级指针pphead存储着一级指针变量phead的地址,解引用一次得到phead的地址
   if(*pphead == NULL)
   {
       *pphead = newnode; //将新结点赋给头结点
   }
   else
   {
       //要进行尾插,就要先找到尾结点
       //尾结点的特征:next指针为空指针
       //建立一个新结点,命名为尾结点
       SLTNode * tail = *pphead;
       while(tail->next != NULL)//遍历找到尾结点
       {
           tail = tail->next;
       }
       tail->next = newnode;
   }
}


线性表之单链表(下)

目录
相关文章
|
9月前
|
存储 监控 安全
企业上网监控系统中红黑树数据结构的 Python 算法实现与应用研究
企业上网监控系统需高效处理海量数据,传统数据结构存在性能瓶颈。红黑树通过自平衡机制,确保查找、插入、删除操作的时间复杂度稳定在 O(log n),适用于网络记录存储、设备信息维护及安全事件排序等场景。本文分析红黑树的理论基础、应用场景及 Python 实现,并探讨其在企业监控系统中的实践价值,提升系统性能与稳定性。
564 1
|
9月前
|
存储 监控 算法
基于跳表数据结构的企业局域网监控异常连接实时检测 C++ 算法研究
跳表(Skip List)是一种基于概率的数据结构,适用于企业局域网监控中海量连接记录的高效处理。其通过多层索引机制实现快速查找、插入和删除操作,时间复杂度为 $O(\log n)$,优于链表和平衡树。跳表在异常连接识别、黑名单管理和历史记录溯源等场景中表现出色,具备实现简单、支持范围查询等优势,是企业网络监控中动态数据管理的理想选择。
242 0
|
算法 数据可视化 开发者
为什么要学习数据结构与算法
今天,我向大家介绍一门非常重要的课程——《数据结构与算法》。这门课不仅是计算机学科的核心,更是每一位开发者从“小白”迈向“高手”的必经之路。
为什么要学习数据结构与算法
|
算法 Java
算法系列之数据结构-Huffman树
Huffman树(哈夫曼树)又称最优二叉树,是一种带权路径长度最短的二叉树,常用于信息传输、数据压缩等方面。它的构造基于字符出现的频率,通过将频率较低的字符组合在一起,最终形成一棵树。在Huffman树中,每个叶节点代表一个字符,而每个字符的编码则是从根节点到叶节点的路径所对应的二进制序列。
402 3
 算法系列之数据结构-Huffman树
|
算法 Java
算法系列之数据结构-二叉搜索树
二叉查找树(Binary Search Tree,简称BST)是一种常用的数据结构,它能够高效地进行查找、插入和删除操作。二叉查找树的特点是,对于树中的每个节点,其左子树中的所有节点都小于该节点,而右子树中的所有节点都大于该节点。
555 22
|
存储 算法
非递归实现后序遍历时,如何避免栈溢出?
后序遍历的递归实现和非递归实现各有优缺点,在实际应用中需要根据具体的问题需求、二叉树的特点以及性能和空间的限制等因素来选择合适的实现方式。
388 59
|
10月前
|
编译器 C语言 C++
栈区的非法访问导致的死循环(x64)
这段内容主要分析了一段C语言代码在VS2022中形成死循环的原因,涉及栈区内存布局和数组越界问题。代码中`arr[15]`越界访问,修改了变量`i`的值,导致`for`循环条件始终为真,形成死循环。原因是VS2022栈区从低地址到高地址分配内存,`arr`数组与`i`相邻,`arr[15]`恰好覆盖`i`的地址。而在VS2019中,栈区先分配高地址再分配低地址,因此相同代码表现不同。这说明编译器对栈区内存分配顺序的实现差异会导致程序行为不一致,需避免数组越界以确保代码健壮性。
218 0
栈区的非法访问导致的死循环(x64)
|
存储 C语言 C++
【C++数据结构——栈与队列】顺序栈的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现顺序栈的基本运算。开始你的任务吧,祝你成功!​ 相关知识 初始化栈 销毁栈 判断栈是否为空 进栈 出栈 取栈顶元素 1.初始化栈 概念:初始化栈是为栈的使用做准备,包括分配内存空间(如果是动态分配)和设置栈的初始状态。栈有顺序栈和链式栈两种常见形式。对于顺序栈,通常需要定义一个数组来存储栈元素,并设置一个变量来记录栈顶位置;对于链式栈,需要定义节点结构,包含数据域和指针域,同时初始化栈顶指针。 示例(顺序栈): 以下是一个简单的顺序栈初始化示例,假设用C语言实现,栈中存储
882 77
|
10月前
232.用栈实现队列,225. 用队列实现栈
在232题中,通过两个栈(`stIn`和`stOut`)模拟队列的先入先出(FIFO)行为。`push`操作将元素压入`stIn`,`pop`和`peek`操作则通过将`stIn`的元素转移到`stOut`来实现队列的顺序访问。 225题则是利用单个队列(`que`)模拟栈的后入先出(LIFO)特性。通过多次调整队列头部元素的位置,确保弹出顺序符合栈的要求。`top`操作直接返回队列尾部元素,`empty`判断队列是否为空。 两题均仅使用基础数据结构操作,展示了栈与队列之间的转换逻辑。
下一篇
开通oss服务