【数据结构】- 链表之单链表(上)

简介: 数据结构学习第四弹——链表之单链表(上)

前言


“偶尔失意 是为了压住翘起的尾巴”

本章是关于数据结构中的链表之单链表(上)

提示:以下是本篇文章正文内容,下面案例可供参考

一、链表


1.1链表的概念及结构


概念:链表是一种物理存储结构上非连续、非顺序的存储结构,数据元素的逻辑顺序是通过链表中的指针链接次序实现的

1.2链表的分类


实际中链表的结构非常多样,以下情况组合起来就有8种链表结构:

单向或者双向

带头或者不带头

循环或者非循环

虽然有这么多的链表的结构,但是我们实际中最常用还是两种结构:

1.无头单向非循环链表:结构简单,一般不会单独用来存数据。实际中更多是作为其他数据结构的子结构,如哈希桶、图的邻接表等等。另外这种结构在笔试面试中出现很多。

2.带头双向循环链表:结构最复杂,一般用在单独存储数据。实际中使用的链表数据结构,都是带头双向循环链表。另外这个结构虽然结构复杂,但是使用代码实现以后会发现结构会带来很多优势,实现反而简单了,后面我们代码实现了就知道了

二、单链表(上)


2.1单链表的实现


//SList.h
#include<stdio.h>
typedef int SLTDataType;
typedef struct SListNode
{
  SLTDataType data;
  struct SListNode* next;
}SLTNode;
void SLTPrint(SLTNode* ps);
//SList.c
#include"SList.h"
void SLTPrint(SLTNode* phead)
{
  SLTNode* cur = phead;
  while (cur != NULL)
  {
    printf("%d->", cur->data);
    cur = cur->next;
  }
  printf("NULL\n");
}

2.2单链表实现的两种结构解析


2.3单链表的接口实现


2.3.1头插


void SLPushFornt(SLTNode* phead, SLTDataType x)
{
  SLTNode* newnode = (SLTNode*)malloc(sizeof(SLTNode));
  if (newnode == NULL)
  {
    perror("malloc fail");
    return;
  }
  newnode->data = x;
  newnode->next = NULL;
  newnode->next  = phead;
  phead = newnode;
}

逻辑分析:

物理分析:

注意:

上述代码中newnode是一个局部变量,phead是一个形参出了作用域就会被销毁所以我们需要在外面留一个指针来找到这个节点的地址

2.3.2温馨提醒 宝子~


当我们在外面留下一个指针来找这个节点并打印的时候我们发现打印出来是个NULL,这是怎么回事呢?

举个例子:

#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
void Swap(int* p1, int* p2)
{
  int tmp = *p1;
  *p1 = *p2;
  *p2 = tmp;
}
void Swap1(int** pp1, int** pp2)
{
  int* tmp = *pp1;
  *pp1 = *pp2;
  *pp2 = tmp;
}
int main()
{
  int a = 0;
  int b = 1;
  Swap(&a, &b);
  printf("a=%d,b=%d\n", a, b);
  int c = 2;
  int d = 8;
  int* px = &c;
  int* py = &d;
  printf("px=%p,py=%p\n", px, py);
  Swap1(&px, &py);
  printf("px=%p,py=%p\n", px, py);
  return 0;
}

2.3.3头插完整版代码


分为三个模块分别是SList.h SList.c Test.c

//SList.h
#include<stdio.h>
#include<stdlib.h>
typedef int SLTDataType;
typedef struct SListNode
{
  SLTDataType data;
  struct SListNode* next;
}SLTNode;
void SLTPrint(SLTNode* phead);
void SLPushFornt(SLTNode** pphead, SLTDataType x);
//SList.c
#include"SList.h"
void SLTPrint(SLTNode* phead)
{
  SLTNode* cur = phead;
  while (cur != NULL)
  {
    printf("%d->", cur->data);
    cur = cur->next;
  }
  printf("NULL\n");
}
SLTNode* BuyTNode(SLTDataType x)//因为头插尾插等都需要maolloc一块新空间所以我们分装成一个函数
{
  SLTNode* newnode = (SLTNode*)malloc(sizeof(SLTNode));
  if (newnode == NULL)
  {
    perror("malloc fail");
    return;
  }
  newnode->data = x;
  newnode->next = NULL;
  return newnode;
}
void SLPushFornt(SLTNode** pphead, SLTDataType x)
{
  SLTNode* newnode = BuyTNode(x);
  newnode->next  = *pphead;
  *pphead = newnode;
}
//Test.c
#include"SList.h"
void TestSList()
{
  SLTNode* plist = NULL;
  SLPushFornt(&plist, 1);
  SLPushFornt(&plist, 2);
  SLPushFornt(&plist, 3);
  SLPushFornt(&plist, 4);
  SLTPrint(plist);
}
int main()
{
  TestSList();
  return 0;
}

2.3.4尾插


void SLPushBack(SLTNode* phead, SLTDataType x)
{
  SLTNode* tail = phead;
  while (tail->next != NULL)
  {
    tail = tail->next;
  }
  SLTNode* newnode = BuyTNode(x);
  tail->next = newnode;
}

2.3.5温馨提醒 宝子~


(1).下面这段代码存在两个问题:


内存泄露

链表并没有链接起来

void SLPushBack(SLTNode* phead, SLTDataType x)
{
  SLTNode* tail = phead;
  while (tail->next != NULL)
  {
    tail = tail->next;
  }
  SLTNode* newnode = BuyTNode(x);
  tail->next = newnode;
}

(2).下面这段代码存在问题:


我们通过运行代码可以知道,尾插不用二级指针也是可以的

但是我们需要考虑两个问题:

尾插时前面是非空链表:

尾插时前面是空链表:

(1). 使用二级指针

 (2). 使用一级指针

总结图:所以总而言之为了满足两者的条件我们使用二级指针

其实就类似于头插,我们头插时并不是开始就有数字开始也是NULL,所以使用二级指针(传址) * pphead指向plist,然后开辟空间newnode赋给*pphead,然后链接,第一次头插完成后, * pphead和newnode销毁,但是plist已经通过传地址改变了里面的地址指向了第一个节点空间地址,当第二次插入数字时,把plist传址给 * pphead;然后开辟了一块新空间newnode,注意哦plist经过上次已经指向了第一个节点而把plist传址给 *pphead这时 *pphead就指向第一个节点的地址只需要让 *pphead赋给头插的那个newnode->next就链接起来了然后 *pphead = newnode,这样出了作用域都销毁而plist也就指向了头插的那个节点的位置;

但是不使用二级指针,只是把plist中的地址赋给phead,phead改变并不影响plist因为出了作用域就会销毁

2.3.6总而言之


要改变int,传int的地址

要改变int * ,传int * 的地址

要改变结构体,传结构体的地址

要改变结构体指针,传结构体的指针的地址

总结


Ending,今天的链表之单链表(上)的内容就到此结束啦~,如果后续想了解更多,就请关注我吧。

相关文章
|
2月前
|
存储 算法 Perl
数据结构实验之链表
本实验旨在掌握线性表中元素的前驱、后续概念及链表的建立、插入、删除等算法,并分析时间复杂度,理解链表特点。实验内容包括循环链表应用(约瑟夫回环问题)、删除单链表中重复节点及双向循环链表的设计与实现。通过编程实践,加深对链表数据结构的理解和应用能力。
67 4
|
5天前
|
机器学习/深度学习 存储 C++
【C++数据结构——线性表】单链表的基本运算(头歌实践教学平台习题)【合集】
本内容介绍了单链表的基本运算任务,涵盖线性表的基本概念、初始化、销毁、判定是否为空表、求长度、输出、求元素值、按元素值查找、插入和删除数据元素等操作。通过C++代码示例详细解释了顺序表和链表的实现方法,并提供了测试说明、通 - **任务描述**:实现单链表的基本运算。 - **相关知识**:包括线性表的概念、初始化、销毁、判断空表、求长度、输出、求元素值、查找、插入和删除等操作。 - **测试说明**:平台会对你编写的代码进行测试,提供测试输入和预期输出。 - **通关代码**:给出了完整的C++代码实现。 - **测试结果**:展示了测试通过后的预期输出结果。 开始你的任务吧,祝你成功!
22 5
|
19天前
|
数据库
数据结构中二叉树,哈希表,顺序表,链表的比较补充
二叉搜索树,哈希表,顺序表,链表的特点的比较
数据结构中二叉树,哈希表,顺序表,链表的比较补充
|
2月前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
80 5
|
2月前
|
存储 C语言
【数据结构】手把手教你单链表(c语言)(附源码)
本文介绍了单链表的基本概念、结构定义及其实现方法。单链表是一种内存地址不连续但逻辑顺序连续的数据结构,每个节点包含数据域和指针域。文章详细讲解了单链表的常见操作,如头插、尾插、头删、尾删、查找、指定位置插入和删除等,并提供了完整的C语言代码示例。通过学习单链表,可以更好地理解数据结构的底层逻辑,提高编程能力。
130 4
|
2月前
|
算法 安全 搜索推荐
2024重生之回溯数据结构与算法系列学习之单双链表精题详解(9)【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构王道第2.3章之IKUN和I原达人之数据结构与算法系列学习x单双链表精题详解、数据结构、C++、排序算法、java、动态规划你个小黑子;这都学不会;能不能不要给我家鸽鸽丢脸啊~除了会黑我家鸽鸽还会干嘛?!!!
|
2月前
|
存储 Web App开发 算法
2024重生之回溯数据结构与算法系列学习之单双链表【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构之单双链表按位、值查找;[前后]插入;删除指定节点;求表长、静态链表等代码及具体思路详解步骤;举例说明、注意点及常见报错问题所对应的解决方法
|
2月前
|
算法
数据结构之购物车系统(链表和栈)
本文介绍了基于链表和栈的购物车系统的设计与实现。该系统通过命令行界面提供商品管理、购物车查看、结算等功能,支持用户便捷地管理购物清单。核心代码定义了商品、购物车商品节点和购物车的数据结构,并实现了添加、删除商品、查看购物车内容及结算等操作。算法分析显示,系统在处理小规模购物车时表现良好,但在大规模购物车操作下可能存在性能瓶颈。
58 0
|
2月前
|
C语言
【数据结构】双向带头循环链表(c语言)(附源码)
本文介绍了双向带头循环链表的概念和实现。双向带头循环链表具有三个关键点:双向、带头和循环。与单链表相比,它的头插、尾插、头删、尾删等操作的时间复杂度均为O(1),提高了运行效率。文章详细讲解了链表的结构定义、方法声明和实现,包括创建新节点、初始化、打印、判断是否为空、插入和删除节点等操作。最后提供了完整的代码示例。
87 0
|
3月前
|
存储
[数据结构] -- 双向循环链表
[数据结构] -- 双向循环链表
30 0