线性表的链式存储结构的特点就是用一组任意的存储单元存储线性表的数据元素,这组存储单元可以在内存中未被占用的任意位置。比起顺序存储结构每个元素只需要存储一个位置就可以了。现在链式存储结构中,除了要存储数据信息外,还要存储它的后继元素的存储地址(指针)。也就是说除了存储其本身的信息外,还需要存储一个指示其直接后继的存储位置的信息。来几个概念:把存储数据元素信息的域称为数据域,把存储直接后继位置的域称为指针域。指针域中存储的信息称为指针或链。这两部分信息组成数据元素称为存储映像,称为结点(Node).01.单链表

单链表

因为此类链表的每个结点中只包含一个指针域,所以叫做单链表对于线性表来说,总得有个头有个尾,链表也不例外。我们把链表中的第一个结点的存储位置叫做头指针,最后一个结点指针为空(null)。头指针头节点的异同:

  • 头节点的数据域一般不存储任何信息,谁叫它是第一个呢,有这个特权。

  • 头指针 是指向链表第一个结点的指针,若链表有头节点,则是指向头结点的指针。

  • 头指针具有标识作用,所以常用的头指针冠以链表的名字(指针变量的名字)。

  • 无论链表是否为空,头指针均不为空

  • 头指针是链表的必要元素。

头结点:

  • 头结点是为了操作的统一和方便而设立的,放在第一个元素的结点之前,其数据域一般无意义(但也可以用来存放链表的长度)。

  • 有了头结点,对在第一个元素结点前插入结点和删除第一结点起操作与其他节点的操作就统一了。

  • 头结点不一定是链表的必须要素。

单链表图例:

空链表图例:

typedef struct Node{    ElemType data; //数据域    struct Node* Next;  //指针域} Node;typedef struct Node* LinkList;

02.

单链表常见操作

单链表的读取
  • 声明一个结点p指向链表第一个结点,初始化j从1开始。
  • 当 j < i 时, 就遍历链表,让 p 的指针向后移动,不断指向下一个结点,j + 1;
  • 若到链表末尾 p 为空,则说明第 i 个元素不存在;否则查找成功,返回结点 p 的数据。
/*GetElem.c*/Status GetElem(LinkList L, int i, ElemType *e) {    int j;    LinkList p;

    p = L->next;    j = 1;

    while( p && j < i){        p = p->next;        j++;    }

    if( !p){        return ERROR;    }

    *e = p->data;

    return OK;}

单链表的读取,说白了,就是从头结点开始找呀找,直到第 i 个元素为止。由于这个算法的时间复杂度取决于 i 的位置,当 i = 1 时, 则不需要遍历,而 i = n 时则遍历n-1次才可以。因此最坏情况的时间复杂度为O(n).由于单链表的结构中没有定义表长,所以不能实现知道循环多少次,因此也就不方便使用for来控制循环,而选择while语句。其核心思想叫做 “工作指针后移”,这是很多算法的常用技术。

单链表的插入

假设存储元素 e 的结点为 s,要将结点 s 插入结点p、p->next 之间的逻辑图示:

一定要注意单链表插入元素时指针的修改顺序奥!

单链表第 i 个数据插入结点的算法思路:

  • 声明一个结点p指向链表头结点,初始化 j 从 1 开始

  • 当 j < 1 时,就遍历链表,让 p 的指针向后移动,不断指向下一个结点, j 累加 1;

  • 若到链表末尾 p 为空,则说明第 i 个元素不存在;

  • 否则查找成功,在系统中生成一个空结点 s;

  • 将数据元素e 赋值给s->data

  • 执行插入语句,并返回成功
Status ListInsert(LinkList *L, int i, ElemType e){    int j;    LinkList p, s;

    p =* L; /*链表的头结点*/    j = 1;

    while( p && j < i) /*寻找第i个结点位置*/    {        p = p->next;        j++;    }    if( !p || j > i )    {        return ERROR; /*第 i 个元素不存在*/    }

    s = (LinkList)malloc(sizeof(Node)); /*生成一个新的结点*/    s->data = e; /*将e赋值给结点s的数据域*/

    s->next = p->next; /*插入步骤1*/    p->next = s; /*插入步骤2*/

    return OK;

}
单链表的删除

要实现第二个结点的删除操作,其实就是将它的前继结点的指针绕过第二个节点,直接指向第二个结点的后继结点即可。

Status ListDelete(LinkList *L, int i, ElemType *e){    int j;    LinkList p,q;    p = *L;    j = 1;

    while( p->next && j    {        p = p->next;        ++j;    }    if( !(p->next) || j>i )    {        return ERROR;    }

    q = p->next;    p->next = q->next; /*这两句相当于p->next = p->next->next;*/

    *e = q->data;    free(q);

    return OK;}

对于插入和删除数据越频繁的操作,单链表相比顺序存储结构而言,效率优势就越明显。03.

单链表的创建

单链表的整表创建
  • 对于顺序存储结构的线性表的整表创建,我们可以用数组的初始化来直观理解。

  • 而单链表和顺序存储结构就不一样了,它不像顺序存储结构数据这么集中,它的数据可以分散在内存的各个角落,它的增长也是动态的。

  • 对于每个链表来说,它所占用空间的大小和位置是不需要预先分配划定的,可以根据系统的情况和实际的需求即时生成。

  • 人生就要追求和单链表一样,灵活应变!

创建单链表的过程是一个动态生成链表的过程,从“空表” 的初始状态起,依次建立各元素结点并逐个插入链表。单链表整表创建的算法思路:

  • 声明一个结点p和计数器变量 i

  • 初始化一个空表L;

  • 让 L 的头结点的指针指向NULL, 即建立一个带头结点的单链表;

  • 循环实现后继结点的赋值和插入。

头插法建立单链表
  • 头插法从一个空表开始,生成新结点,读取数据存放到新结点的数据域中,然后将新结点插入到当前链表的表头上,知道结束为止。

  • 简单来说,就是把新加进来的元素放到表头后的第一个位置:

    • 先让新结点的next指针指向头节点之后,然后让表头的next指向新结点

  • 就相当于插队一样,只是头插法让新来的结点插到第一个位置。

void CreateListHead(LinkList *L, int n){    LinkList p;    int i;

    srand(time(0)); //初始化随机数种子

    *L = (LinkList)malloc(sizeof(Node));    (*L)->next = NULL;

    for( i=0; i    {        p = (LinkList)malloc(sizeof(Node)); //生成新结点        p->data = rand()%100+1; //生成1到100之间的随机数        p->next = (*L)->next;        (*L)->next = p;    }}
尾插法建立单链表

头插法建立链表虽然算法简单,但生成的链表中结点的次序和输入的顺序相反。

就像现实社会中我们鄙视插队不遵守纪律的孩子,每次排队打饭时看到在你前面插队的小盆友就想到头插法,在你后面插队的就想到尾插法。编程中把新结点直接插入到最后的方式建立单链表就称为尾插法。

void CreateListTail(LinkList *L, int n){    LinkList p, r;    int i;    srand(time(0));    *L = (LinkList)malloc(sizeof(Node));    r = *L; //r初始化指向尾部结点

    for( i=0; i        p = (Node *)malloc(sizeof(Node));        p->data = rand()%100+1;        r->next = p;        r = p;    }}

单链表的整表删除

当我们不打算使用单链表时,需要将其销毁。其实就是在内存中将它释放掉,以便留出空间给其他程序或软件使用。

单链表整表删除算法思路:

  • 声明结点p和q;

  • 将第一个结点赋值给p,下一个结点赋值给q;

  • 循环执行失望p和q赋值给p的操作;
Status ClearList(LinkList *L){    LinkList p, q;    p = (*L)->next; //将第一个节点赋值给p

    while(p)    {        q = p->next;        free(p);        p = q;    }

    (*L)->next = NULL;    return OK;}

注意观察上面代码中while循环体内,可能觉着q变量没有存在的必要,只需要在循环体内直接写free(p);p = p->next; 但这样做是不行滴!要知道p是一个结点,它除了数据域,还有指针域。当我们做free(p);的时候,其实是对它整个结点进行删除和内存释放的工作,也就是把p的指针域(指向下一个结点)也给删除了,所以需要变量q来保存p的下一个结点。

单链表结构与顺序存储结构优缺点分析

我们分别从存储分配方式、时间性能、空间性能三方面来做对比。

  1. 存储分配方式:

  • 顺序存储结构用一段连续的存储单元依次存储线性表的数据元素。

  • 单链表采用链式存储结构,用一组任意的存储单元存放线性表的元素。

时间性能:

  • 顺序存储结构需要平均移动表长一半的元素,时间为O(n)

  • 单链表在计算出某位置的指针后,插入和删除时间仅为O(1)

  • 顺序存储结构O(1)

  • 单链表O(n)

  • 查找

  • 插入和删除

空间性能:

  • 顺序存储结构需要预分配存储空间,分大了,容易造成空间浪费,分小了,容易发生溢出。

  • 单链表不需要分配存储空间,只要有就可以分配,元素个数也不受限制。

综上所述,我们得出一些经验性的结论:

  • 若线性表需要频繁查找,很少进行插入和删除操作时,宜采用顺序存储结构

  • 若需要频繁插入和删除时,宜采用单链表结构

  • 比如游戏开发中,对于用户注册的个人信息,除了注册时插入数据外,绝大多数情况都是读取,所以应该考虑用顺序存储结构。

  • 而游戏中玩家的武器或者装备列表,随着玩家的游戏过程中,可能会随时增加或删除,此时再用顺序存储就不太合适了,单链表结构就可以大展拳脚了。

  • 当线性表中的元素个数变化较大或者根本不知道有多大时,最好用单链表结构,这样可以不需要考虑存储空间的大小问题

  • 而如果事先知道线性表的大致长度,比如一年12个月,一周就是星期一至星期天共七天,这种用顺序存储结构效率会更高。

  • 总之,线性表的顺序存储结构和单链表结构各有优缺点,不能简单的说哪个好,那个不好,需要根据实际情况,来综合平衡采用哪种数据结构能达到需要和性能。

04.

线性表回顾

顺序表的插入操作

顺序表的删除操作

单链表的插入操作

你好呀!由于一篇图文只能插入三个视频,但还有三个视频,我给大家放到了第二篇图文中,如果你需要线性表的 flash 演示动画,你可以后台回复 “线性表” 获得两篇图文中的所有的 flash 动画资料。

推荐阅读

LeetCode上稀缺的四道shell编程题解析

号称三剑客之首的awk,开始秀!SHELL编程三剑客之sed命令你想知道的线性表的内容都在这里!!!数据结构与算法基础

END

作者:景禹,一个追求极致的共享主义者,想带你一起拥有更美好的生活,化作你的一把伞。

python 单链表节点怎么快速定义_线性表链式存储结构之单链表相关推荐

  1. Linux从入门到精通系列之线性表链式存储结构-单链表原理解析

    前言 线性表的链式存储结构的特点就是用一组任意的存储单元存储线性表的数据元素,这组存储单元可以在内存中未被占用的任意位置. 比起顺序存储结构每个元素只需要存储一个位置就可以了.现在链式存储结构中,除了 ...

  2. [一] 详细讲解: 线性表链式存储结构 中的 单链表; (数据结构和算法)

    一.定义 二.单链表

  3. 线性表的链式存储结构以及单链表的插入和删除原理实现

    线性表的链式存储结构 线性表中的每个元素最多只有一个前驱元素和一个后继元素(其逻辑结构),因此可以采用链式存储结构存储. 链表 线性表的链式存储结构称为链表.在链表中每个结点不仅包含有元素本身的信息( ...

  4. 数据结构--线性表链式存储(链表)--单链表

    定义: 单链表是一种链式存取的数据结构,用一组地址任意的存储单元存放线性表中的数据元素. 链表中的数据是以节点来表示的,每个结点的构成:元素( 数据元素的映象) + 指针(指示后继元素存储位置),元素 ...

  5. php数据结构链表代码,数据结构之线性表——链式存储结构之单链表(php代码实现)...

    /** * * 1. 类LNode用作创建单链表时,生成新的节点. * 2. 类SingleLinkList用于创建单链表以及对单链表的一些操作方法(实例化此类就相当于创建了一个空链表) * 3. C ...

  6. 数据结构之线性表——链式存储结构之单链表(php代码实现)

    <?php /**** 1. 类LNode用作创建单链表时,生成新的节点.* 2. 类SingleLinkList用于创建单链表以及对单链表的一些操作方法(实例化此类就相当于创建了一个空链表)* ...

  7. 栈的链式存储结构(企业级链表)

    #include<stdio.h> #include<stdlib.h> #include<string.h> struct StackNode{struct St ...

  8. 二叉树的链式存储结构--二叉链表

    1 二叉树的链式存储结构 //二叉链表的结点结构定义typedef int TElemType; typedef struct BiTNode {TElemType data;struct BiTNo ...

  9. 链表的特点,单链表的定义、存储结构,单链表的基本操作(判断链表是否为空、销毁链表、清空链表、求链表表长、查找、插入、删除,建立单链表)

    目录 一.链表(链式存储结构)的特点 二.单链表的定义和表示 1.带头结点的单链表 2.单链表的存储结构 三.单链表基本操作的实现 1.单链表的初始化(带头结点的单链表) 2.补充单链表的几个常用简单 ...

最新文章

  1. TensorFlow:Object_Detection_API在Windows10上的配置
  2. 转学美本半年,我眼里的中美高等教育
  3. 无限级菜单 mysql设计_无限级菜单简单的设计
  4. oracle内部函数,[数据库]Oracle内置函数
  5. 路径处理库pathlib使用详解
  6. 离散化+树状数组求逆序数
  7. tstringlist怎么查看是否存在该数据_财务报表审计该如何进行?
  8. python中文字符串转list
  9. ASP.NET案例--新闻模块设计
  10. java mysql 分页计算公式_关于Java的分页算法,急!
  11. ssh远程登录报错:WARNING: REMOTE HOST IDENTIFICATION HAS CHANGED
  12. SQL Server添加Check约束
  13. 为什么html字体之间有间隔,css字体间隔怎么调整?css文字间隔的调整方法
  14. Module ‘xxx‘ was created for incompatible target arm64-apple-ios9.0 Xcode12 错误解决
  15. SyntaxError: invalid syntax解决方案
  16. 苹果系统安装Java开发环境JDK
  17. 使用安卓原生系统刷机,修改
  18. Vue | 指令实现自动填充英文名功能
  19. C语言中system函数用法解释
  20. 中标麒麟编译qgis源码+PyQt环境

热门文章

  1. python数据可视化雷达图程序_Python数据可视化之matplotlib
  2. PostgreSQL体系结构和基本操作
  3. 【现场福利+直播通道】2020数据技术嘉年华正式“菜单”出炉,饕餮盛宴等你共享!...
  4. 下班约会时来了新需求,咋办?
  5. 教你用Python自制拼图小游戏,轻松搞定熊孩子
  6. Linux神器strace的使用方法及实践
  7. 【华为云技术分享】Linux内核编程环境 (2)
  8. 【Python成长之路】装逼的一行代码:快速共享文件【华为云分享】
  9. Docker集群管理之Swarm介绍
  10. iphone7测试软件,iPhone7如何测试网速 ping命令测试网速方法介绍