当前位置:   article > 正文

数据结构与算法——线性表篇详解

数据结构与算法——线性表篇详解

1. 线性表

1.1 概述

  • 线性表:是一种最常用、最简单,也是最基本的数据结构。
  • 线性表由n个数据元素所构成的有限序列,且数据类型相同。
  • 线性表可以用顺序存储链式存储两种存储结构来表示。
    • 使用顺序存储的线性表称为顺序表,也称为静态线性表。
    • 使用链式存储的线性表称为链表,也称为动态线性表。
  • 链表的分类:单链表、双向链表、循环链表。

1.2 顺序表

1.2.1 定义

  • 顺序表,就是顺序存储的线性表。

  • 顺序存储是用一组地址连续的存储单元依次存放线性表中各个数据元素的存储结构。

    • 在逻辑上,数据ABCD是连续

    • 在物理上,地址也是连续的

  • 可以使用数组来描述数据结构中的顺序存储结构。

1.2.2 地址公式

  • 地址公式

    1. //第i的地址 = 第一个地址 + 第几个 * 存储单位
    2. Loc(ai) = Loc(a0) + i * c

1.2.3 顺序表特点

  1. 在线性表中逻辑上相邻的数据元素,在物理存储位置上也是相邻的。

  2. 存储密度高。但需要预先分配“足够”的存储空间。

    存储密度 = 数据元素存储空间 / 数据元素实际占用空间

    在顺序表中,存储密度为1。

  3. 便于随机存储。(数组中可以通过下标进行存储)

  4. 不便于插入和删除操作。两种操作都会引起大量的数据移动。

1.2.4 算法:插入

  • 需要:在顺序表第i个位置处插入一个新元素。

  • 顺序表插入操作:将第i个数据元素及其之后的所有的数据元素,后移一个存储位置,再将新元素插入到i处。

  • 前置:类中成员

    1. public class SqList {
    2. private Object[] listElem; //线性表的存储空间
    3. private int curLen; //线性表的当前长度
    4. }
  • 插入操作算法

    1. /**
    2. * @Param i 第i个位置
    3. * @Param x 需要插入的数据
    4. */
    5. public void insert(int i, Object x) {
    6. //0.1 满校验:存放实际长度 和 数组长度 一样
    7. if(curLen == listEle.length) {
    8. throw new Exception("已满");
    9. }
    10. //0.2 非法校验,在已有的数据中间插入 [0, curLen],必须连续,中间不能空元素
    11. if(i < 0 || i > curLen) {
    12. throw new Exception("位置非法");
    13. }
    14. //1 将i及其之后后移
    15. for(int j = curLen ; j > i; j --) {
    16. listEle[j] = listEle[j-1];
    17. }
    18. //2 插入i处
    19. listEle[i] = x;
    20. //3 记录长度
    21. curLen ++;
    22. }
  • 插入时间复杂度:O(n)

1.2.5 算法:删除

  • 需求:将第i位置处元素删除
  • 删除操作:将第i个数据元素ai之后的所有数据元素向前一定一个存储位置。

  • 算法:

    1. public void remove(int i ) throws Exception {
    2. // 0.1 校验非法数据
    3. if(i < 0 || i > curLen - 1 ) {
    4. throw new Exception("位置非法");
    5. }
    6. // 1 将i之后向前移动
    7. for(int j = i ; j < curLen - 1 ; j ++ ) {
    8. listEle[j] = listEle[j+1];
    9. }
    10. // 2 长度减一
    11. curLen--;
    12. }
  • 删除时间复杂度:O(n)

1.2.6 算法:查找

  • 需求:查找指定数据的索引号

  • 算法1:循环遍历已有数据,进行判断,如果有返回第一个索引号,如果没有返回-1

    1. public int indexOf(Object x) {
    2. for(int i = 0; i < curLen ; i ++) {
    3. if(listEle[i].equals(x)) {
    4. return i;
    5. }
    6. }
    7. return -1;
    8. }
  • 算法2:使用变量记录没有匹配到索引

    1. public int indexOf(Object x) {
    2. int j = 0; //用于记录索引信息
    3. while(j < curLen && !listElem[j].equals(x)) {
    4. j++;
    5. }
    6. // j记录索引小于数量
    7. if(j < curLen ) {
    8. return j;
    9. } else {
    10. return -1
    11. }
    12. }
  • 查询时间复杂度:O(n)

1.2.7 顺序表局限性:

  1. 若要为线性表扩充存储空间,则需要重新创建一个地址连续的更大的存储空间,并把原有的数据元素都复制到新的存储空间中。(扩容)
  2. 因为顺序存储要求逻辑上相邻的数据元素,在物理存储位置上也是相邻的,这就使得要增删数据元素时,会引起平均一半的数据元素的移动。

1.3 单链表

1.3.1 定义

  • 采用链式存储方式存储的线性表称为链表。

  • 链表中每一个结点包含存放数据元素值的数据域和存放逻辑上相邻节点的指针域。

    • 数据域 data:用于存放数据元素的值。
    • 指针域 next:用于存放后继结点的地址。

1.3.2 术语

  • 非空表:

  • 空表:空表的指针域为空指针null

1.3.3 类定义

  • 数据域 data:用于存放数据元素的值。
  • 指针域 next:用于存放后继结点的地址。
  1. public class Node{
  2. public Object data; //数据域
  3. public Node next; //指针域
  4. }

1.3.4 算法:单链表的长度【重点】

  1. public int length() {
  2. Node p = head.next; // 获得第一个结点
  3. int length = 0; // 定义一个变量记录长度
  4. while(p != null) {
  5. length ++; //计数
  6. p = p.next; //p指向下一个结点
  7. }
  8. return length;
  9. }

1.3.5 算法:按索引号(位序号)查找【重点】

  • 思路:
    • 处理了所有的结点都没有数据,判断依据 null
    • 处理到一部分就找到了,判断依据
  1. public Object get(int i) {
  2. Node p = head.next; //获得头结点
  3. int j = 0; //已经处理的结点数
  4. while(p != null && j < i) { //链表没有下一个 && 处理有效部分
  5. p = p.next;
  6. j++;
  7. }
  8. if(i < 0 || p == null) {
  9. throw new Exception("元素不存在");
  10. }
  11. return p.data; //返回数据
  12. }

1.3.6 算法:按值查找索引号【重点】

  1. //查询给定值的索引号,如果没有返回-1
  2. public int indexOf(Object x) {
  3. Node p = head.next; // 获得头结点
  4. int j = 0; // 不匹配元素的个数
  5. while(p != null && !p.data.equals(x)) { // 循环依次找【不匹配】元素
  6. p = p.next;
  7. j++;
  8. }
  9. // 返回匹配索引,如果没有返回-1
  10. if(p != null) {
  11. return j;
  12. } else {
  13. return -1;
  14. }
  15. }

1.3.7 算法:插入

  • 需求:向索引i前面插入一个新结点s
  • 思路:获得i的前驱,结点P
  1. public void insert(int i , Object x) {
  2. Node p = ...; // 获得i前驱,结点p
  3. Node s = new Node(x); //创建新结点s
  4. s.next = p.next; //必须先执行
  5. p.next = s; //然后再执行
  6. }

1.3.8 算法:删除

  • 需求:删除i结点

  1. public void remove(int i) {
  2. // 获得i的前驱结点p
  3. p.next = p.next.next;
  4. }

1.3.9 算法:获得前驱

  1. // i最小值0,表示获得头结点head
  2. // 算法内部,使用-1表示头结点head,结点移动从头结点head开始。
  3. public void previous(int i) {
  4. Node p = head; //从头结点head开始移动
  5. int j = -1; //使用-1表示头结点的索引
  6. while(p != null && j < i-1 ) { // i-1前驱的下标
  7. p = p.next;
  8. j++;
  9. }
  10. if(p == null || i < 0) { // i-1 < j
  11. throw new RuntimeException("i不合法");
  12. }
  13. //p就是需要获得前驱
  14. }

1.4 循环链表

1.4.1 定义

  • 循环链表也称为环形链表,其结构与单链表相似,只是将单链表的首尾相连。

  1. // p结点是尾结点
  2. p.next = head;

1.4.2 算法:循环链表合并

  • 分析

  • 核心算法
  1. /* 变量
  2. a尾: taila
  3. a头:taila.next
  4. b尾:tailb
  5. b头:tailb.next
  6. */
  7. // 先将b尾指向a的头,需要定义p记录b尾原来的值
  8. // 再将a的尾指向b的头
  9. Node p = tailb.next; //记录b尾的指向,也就是b头
  10. tailb.next = taila.next; //b尾指向a头
  11. taila.next = p.next; //a尾执行b头

1.5 双向链表

1.5.1 定义

  • 一个结点由两个指针域,一个指针域指向前驱结点,另一个指针域指向后继结点,这种链表称为双向链表
    • 数据域 data
    • 前驱结点指针域:prior
    • 后继结点指针域:next

1.5.2 算法:插入

  • 需求:向结点p前面插入新结点s

  1. /* 变量
  2. 结点a:p.prior
  3. */
  4. p.prior.next = s; //结点a指向结点s
  5. s.prior = p.prior; // 结点s指向结点a
  6. p.prior = s; //结点p指向结点s
  7. s.next = p; //结点s执行结点p
  8. //注意:必须先处理结点a,否则无法获得结点a

1.5.3 算法:删除

  • 需求:删除p结点

  1. // a.next = b
  2. p.prior.next = p.next;
  3. // b.prior = a
  4. p.next.prior = p.prior

=========================================================

后记

好啦,以上就是本期全部内容,能看到这里的人呀,都是能人

十年修得同船渡,大家一起点关注。

我是♚焕蓝·未来,感谢各位【能人】的:点赞收藏评论,我们下期见!

各位能人们的支持就是♚焕蓝·未来前进的巨大动力~

注:如果本篇Blog有任何错误和建议,欢迎能人们留言!

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/正经夜光杯/article/detail/918873
推荐阅读
相关标签
  

闽ICP备14008679号