• 顺序表和链表


    作者:~小明学编程 

    文章专栏:Java数据结构

    格言:目之所及皆为回忆,心之所想皆为过往
    在这里插入图片描述

    目录

    顺序表

    概念和结构

    实现顺序表

    接口实现

    打印顺序表

    判断顺序表是否已满

    判断顺序表是否为空

    获取顺序表的长度

    在指定位置添加元素

    判断是否包含某个元素

    获取pos位置的元素

    给 pos 位置的元素设为 value

     删除第一次出现的关键字key

    清空顺序表

    完整的顺序表以及功能方法

    功能演示

    顺序表的问题及思考

    链表

    链表的概念及结构

     链表的实现(无头单项非循环链表)

    链表的单位结构

    创建链表

    打印链表

    插入元素(头插法)

    插入元素(尾插法)

    插入元素(任意位置插入)

    查找是否包含关键字key是否在单链表当中

    删除第一次出现关键字为key的节点

    得到单链表的长度

    清空链表


    顺序表

    概念和结构

    顺序表是用一段 物理地址连续 的存储单元依次存储数据元素的线性结构,一般情况下采用数组存储。在数组上完成数据的增删查改。
    顺序表一般分为两种
    1.静态顺序表:使用定长数组存储。
    2.动态顺序表:使用动态开辟的数组存储。
    静态顺序表适用于确定知道需要存多少数据的场景 .
    静态顺序表的定长数组导致 N定大了,空间开多了浪费,开少了不够用,相比之下动态顺序表更灵活 , 根据需要动态的分配空间大小 .

    实现顺序表

    1. public class MyArrayList {
    2. public int[] elem;
    3. public int numSize;
    4. public MyArrayList() {
    5. this.elem = new int[10];
    6. }
    7. }
    这里我们首先创建一个类,类的里面有elem数组,还有numSize用于记录我们的顺序表里面元素的个数,最后还有一个构造方法用于初始化我们的数组。

    接口实现

    打印顺序表

    1. public void disPlay() {
    2. for (int i = 0; i < numSize; i++) {
    3. System.out.print(elem[i]+" ");
    4. }
    5. System.out.println();
    6. }

    打印顺序表的实现原理很简单就是循环数组遍历里面元素的个数,其中numSize就是顺序表里面元素的个数。

    判断顺序表是否已满

    1. //判断顺序表是否是满的
    2. public boolean isFull() {
    3. if(this.elem.length==this.numSize) {
    4. return true;
    5. }else {
    6. return false;
    7. }
    8. }

    当我们的数组长度和我们放进去的数组个数相等的时候就代表着我们的顺序表已经满了。

    判断顺序表是否为空

    1. //判断顺序表是否为空
    2. public boolean isEmpty(){
    3. return numSize==0;
    4. }

    此方法也比较简单就是当顺序表为空也就是numSize的值是0的时候顺序表就为空然后直接返回

    numSize==0的布尔值。

    获取顺序表的长度

    1. //获取顺序表长度
    2. public int size(){
    3. return this.numSize;
    4. }

    想要获取顺序的长度的话就直接返回当前顺序的元素的个数就行了。

    在指定位置添加元素

    1. //在pos位置增加一个元素
    2. public void add(int pos,int data) {
    3. if (pos<0||pos>numSize){
    4. System.out.println("您添加的位置不合法!");
    5. }else {
    6. if(isFull()){//元素已满进行扩容
    7. this.elem = Arrays.copyOf(this.elem,this.elem.length+2);
    8. }
    9. for (int i = numSize-1; i >= pos ; i--) {
    10. this.elem[i+1] = this.elem[i];
    11. }
    12. this.elem[pos] = data;
    13. this.numSize++;
    14. }
    15. }

    首先我们要判断插入的位置是否是合法的,当pos小于0,或者pos的值大于以后的个数都是不合法的。

    当位置合法的之后我们还得考虑我们插入的时候顺序表是否已经满了,如果已经满了我们则需要对其进行扩容,扩容的话就用到我们的copyOf函数就行了,我们这里扩容的时候采用的是每次扩两个。

    以上都处理完毕后就可以开始我们的插入操作了,插入的时候我们必须从后面开始,如果从前开始的话,下一个元素就会被覆盖掉。

    如图所示我们先将5移动到下一个位置,再将4移动到5的位置,再将3移动到4的位置,最后将我们插入的8放在原本3的位置,这样就完成了我们的插入操作了。

    判断是否包含某个元素

    1. //判断是否包含某个元素
    2. public boolean contains(int toFind) {
    3. for (int i = 0; i < numSize; i++) {
    4. if(this.elem[i]==toFind) {
    5. return true;
    6. }
    7. }
    8. return false;
    9. }

     判断是否包含某个元素的话就直接遍历整个顺序表,找到与目标元素相同的元素的时候就可以直接返回true,当我们遍历完整个顺序表都没有找到对应的元素就返回false表示没有找到。

    获取pos位置的元素

    1. // 获取 pos 位置的元素
    2. public int getPos(int pos){
    3. if(pos<0||pos>numSize){
    4. return -1;
    5. }
    6. if(isEmpty()){
    7. return -1;
    8. }
    9. return this.elem[pos];
    10. }

    这个方法主要就是异常的处理,当我们想要获取的位置不合理,或者我们的顺序表是个空的顺序表的时候,我们就直接返回我们的-1,就相当于我们的异常,处理好这些异常之后我们就可以直接返回对应位置的元素了。

    给 pos 位置的元素设为 value

    1. // 给 pos 位置的元素设为 value
    2. public void setPos(int pos, int value){
    3. if(pos<0||pos>numSize){
    4. System.out.println("pos位置不合法!");
    5. }
    6. if(isEmpty()){
    7. System.out.println("顺序表为空!");
    8. }
    9. this.elem[pos] = value;
    10. }

    此方法的实现和获取pos位置的元素实现原理如出一辙,处理完异常之后就直接赋值就可以了。

     删除第一次出现的关键字key

    1. //删除第一次出现的关键字key
    2. public void remove(int toRemove) {
    3. if(getPos(toRemove)==-1){
    4. System.out.println("要删除的元素不存在");
    5. }else {
    6. for (int i = getPos(toRemove); i < numSize-1; i++) {
    7. this.elem[i] = this.elem[i+1];
    8. }
    9. numSize--;
    10. }
    11. }

    首先处于严谨我们先确定一下要删除的关键字是否存在,判断存在后我们先找到我们要删除元素的位置,接着我们从前向后依次赋值将想要删除的元素覆盖,最后将我们的元素个数减一。

    清空顺序表

    1. //清空顺序表
    2. public void clear(){
    3. numSize = 0;
    4. }

    完整的顺序表以及功能方法

    1. //打印顺序表
    2. public void disPlay() {
    3. for (int i = 0; i < numSize; i++) {
    4. System.out.print(elem[i]+" ");
    5. }
    6. System.out.println();
    7. }
    8. //在pos位置增加一个元素
    9. public void add(int pos,int data) {
    10. if (pos<0||pos>numSize){
    11. System.out.println("您添加的位置不合法!");
    12. }else {
    13. if(isFull()){//元素已满进行扩容
    14. this.elem = Arrays.copyOf(this.elem,this.elem.length+2);
    15. }
    16. for (int i = numSize-1; i >= pos ; i--) {
    17. this.elem[i+1] = this.elem[i];
    18. }
    19. this.elem[pos] = data;
    20. this.numSize++;
    21. }
    22. }
    23. //判断顺序表是否是满的
    24. public boolean isFull() {
    25. if(this.elem.length==this.numSize) {
    26. return true;
    27. }else {
    28. return false;
    29. }
    30. }
    31. //判断是否包含某个元素
    32. public boolean contains(int toFind) {
    33. for (int i = 0; i < numSize; i++) {
    34. if(this.elem[i]==toFind) {
    35. return true;
    36. }
    37. }
    38. return false;
    39. }
    40. // 获取 pos 位置的元素
    41. public int getPos(int pos){
    42. if(pos<0||pos>numSize){
    43. return -1;
    44. }
    45. if(isEmpty()){
    46. return -1;
    47. }
    48. return this.elem[pos];
    49. }
    50. //判断顺序表是否为空
    51. public boolean isEmpty(){
    52. return numSize==0;
    53. }
    54. // 给 pos 位置的元素设为 value
    55. public void setPos(int pos, int value){
    56. if(pos<0||pos>numSize){
    57. System.out.println("pos位置不合法!");
    58. }
    59. if(isEmpty()){
    60. System.out.println("顺序表为空!");
    61. }
    62. this.elem[pos] = value;
    63. }
    64. //删除第一次出现的关键字key
    65. public void remove(int toRemove) {
    66. if(getPos(toRemove)==-1){
    67. System.out.println("要删除的元素不存在");
    68. }else {
    69. for (int i = getPos(toRemove); i < numSize-1; i++) {
    70. this.elem[i] = this.elem[i+1];
    71. }
    72. numSize--;
    73. }
    74. }
    75. //获取顺序表长度
    76. public int size(){
    77. return this.numSize;
    78. }
    79. //清空顺序表
    80. public void clear(){
    81. numSize = 0;
    82. }

    功能演示

    前面我们已经将我们的类以及里面的方法全部写好了,接下来我们就可以测试一下我们的顺序表是否可以使用了。

    1. public class Dome {
    2. public static void main(String[] args) {
    3. MyArrayList myArrayList = new MyArrayList();
    4. for (int i = 0; i < 20; i++) {
    5. myArrayList.add(i,i);
    6. }
    7. myArrayList.add(15,99);
    8. myArrayList.disPlay();
    9. myArrayList.setPos(5,88);
    10. myArrayList.disPlay();
    11. System.out.println(myArrayList.contains(14));
    12. System.out.println(myArrayList.getPos(16));
    13. myArrayList.remove(10);
    14. myArrayList.disPlay();
    15. myArrayList.remove(100);
    16. myArrayList.clear();
    17. myArrayList.disPlay();
    18. }
    19. }

    顺序表的问题及思考

    1. 顺序表中间 / 头部的插入删除,时间复杂度为 O(N)
    2. 增容需要申请新空间,拷贝数据,释放旧空间。会有不小的消耗。
    3. 增容一般是呈 2 倍的增长,势必会有一定的空间浪费。例如当前容量为 100 ,满了以后增容到 200 ,我们再继 续插入了5 个数据,后面没有数据插入了,那么就浪费了 95 个数据空间。

    链表

    链表的概念及结构

    链表是一种 物理存储结构上非连续 存储结构,数据元素的 逻辑顺序 是通过链表中的 引用链接 次序实现的 。
    实际中链表的结构非常多样,以下情况组合起来就有 8 种链表结构:
    单向、双向
    带头、不带头
    循环、非循环

     

     

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

    无头双向链表:在 Java 的集合框架库中 LinkedList 底层实现就是无头双向循环链表。

     链表的实现(无头单项非循环链表)

    链表的单位结构

    1. class ListNode {
    2. public int val;
    3. public ListNode next;
    4. public ListNode(int val) {
    5. this.val = val;
    6. }
    7. }

    这就是组成我们链表的结构我们把它封装成一个类,类里面有存放我们数据的val,还有存放下一个节点的指针next,最后我们再存放一个构造方法方便我们存放数据。

    创建链表

    1. public class MyLinkedList {
    2. public ListNode head;
    3. public void createList() {
    4. ListNode listNode1 = new ListNode(1);
    5. ListNode listNode2 = new ListNode(100);
    6. ListNode listNode3 = new ListNode(32);
    7. ListNode listNode4 = new ListNode(42);
    8. ListNode listNode5 = new ListNode(52);
    9. listNode1.next = listNode2;
    10. listNode2.next = listNode3;
    11. listNode3.next = listNode4;
    12. listNode4.next = listNode5;
    13. this.head = listNode1;
    14. }
    15. }

    我们这里再次封装一个大类里面放着我们对链表的各种操作,其中我们首先介绍一下我们创建链表的方法,这里说明一下最开始的引用变量head,head是用来存放我们第一个节点的地址的。

    接着我们通过创建对象的方法来创建出我们的单位链表,然后再将它们链接起来就组成了我们的链表。

    打印链表

    1. //打印所有节点里面的值
    2. public void display() {
    3. ListNode cur = this.head;
    4. while(cur!=null) {
    5. System.out.print(cur.val+" ");
    6. cur = cur.next;
    7. }
    8. System.out.println();
    9. }

    我们首先定义一个首节点的引用,然后遍历我们整个链表,如果我们的cur所引用的对象的地址是一个空地址就代表着,遍历到了我们的最后一个元素了,至此打印结束。

    插入元素(头插法)

    1. //头插法
    2. public void addFirst(int data) {
    3. ListNode node = new ListNode(data);
    4. node.next = head;
    5. head = node;
    6. }

    我们先new一个新的对象用于存放我们的新元素,然后将我们新对象的所指向的下一个节点改为指向原来的head也就是首节点,最后我们再将我们的首节点指向我们的新对象。

    插入元素(尾插法)

    1. //尾插法
    2. public void addLast(int data) {
    3. ListNode node = new ListNode(data);
    4. ListNode cur = this.head;
    5. if(cur==null) {
    6. head = node;
    7. } else {
    8. while(cur.next!=null) {
    9. cur = cur.next;
    10. }
    11. cur.next = node;
    12. }
    13. }

    首先,我们要new一个对象并且复制一个首元素进行操作,然后分两种情况讨论,一种就是空链表,另外就是非空,当我们的链表是空的时候我们直接将首节点改为node就行了,当非空时通过一个循环找到最后一个元素,然后将最后一个元素的指针改为我们的node就行了。

    插入元素(任意位置插入)

    1. //任意位置插入,第一个数据节点为0号下标
    2. public void addIndex(int index,int data) {
    3. ListNode node = new ListNode(data);
    4. ListNode cur = this.head;
    5. //插入位置是0时直接头插
    6. if(index==0) {
    7. addFirst(data);
    8. return;
    9. }
    10. //插入位置是最后的时候直接是尾插
    11. if (index==size()) {
    12. addLast(data);
    13. return;
    14. }
    15. while(index-1!=0&&cur!=null) {
    16. cur = cur.next;
    17. index--;
    18. }
    19. if (cur==null){
    20. System.out.println("插入位置不合法!");
    21. return;
    22. }
    23. node.next = cur.next;
    24. cur.next = node;
    25. }

    首先,我们要new一个对象并且复制一个首元素进行操作,因为我们要实现从任意位置插入的方法,所以我们插入的方式就可能有三种情况,分别是头插,尾插,还有中间插入,前面两种情况我们前面已经实现了,只需要判断一下相应的情况然后调用该方法就可以了,对于从中间插入的情况我们再做分析,首先就是要判断插入位置的合理性index下标过大或者过小的话都会导致我们的cur最后指向null进而告诉我们插入位置不合理,当满足我们的插入条件后我们就可以进行插入操作了。

     我们新节点node的next是我们插入节点的下一个节点的地址,也就是对应我们的

    node.next = cur.next;

    然后插入位置的的cur的next则是我们插入元素的地址,也就是

    cur.next = node;

    查找是否包含关键字key是否在单链表当中

    1. //查找是否包含关键字key是否在单链表当中
    2. public boolean contains(int key) {
    3. ListNode cur = this.head;
    4. while(cur!=null) {
    5. if(cur.val==key) {
    6. return true;
    7. }
    8. cur = cur.next;
    9. }
    10. return false;
    11. }

    这个太简单了不解释。

    删除第一次出现关键字为key的节点

    1. //删除第一次出现关键字为key的节点
    2. public void remove(int key) {
    3. if(this.head==null) {
    4. System.out.println("该链表为空!");
    5. }else {
    6. ListNode cur = this.head;
    7. if (cur.val==key) {
    8. this.head = cur.next;
    9. }else {
    10. while(cur.next!=null){
    11. if(cur.next.val==key){
    12. cur.next = cur.next.next;
    13. break;
    14. }
    15. cur = cur.next;
    16. if (cur.next==null) {
    17. System.out.println("没找到要删除的结点!");
    18. }
    19. }
    20. }
    21. }
    22. }

    首先还是判断链表是否为空和复制一下我们的首节点,接着判断一下我们的首节点是不是我们要删除的节点是的话我们的首节点就直接指向我们的下一个节点,不是的话我们就通过循环找到我们的key,因为我们找到的是cur.next.val==key也就是我们目标的上一个节点,所以我们就将上一个节点,直接指向我们的下一个节点就可以了,没找到key的话就最终指向null,然后输出相关的语句。

    1. //删除所有值为key的节点
    2. public ListNode removeAllKey(int key) {
    3. if(this.head==null) {
    4. System.out.println("该链表为空!");
    5. return null;
    6. }else {
    7. ListNode cur = this.head.next;
    8. ListNode prev = this.head;
    9. while (cur!=null) {
    10. if (cur.val == key) {
    11. prev.next = cur.next;
    12. cur = cur.next;
    13. } else {
    14. prev = cur;
    15. cur = cur.next;
    16. }
    17. }
    18. if (this.head.val==key) {
    19. this.head = this.head.next;
    20. }
    21. return this.head;
    22. }
    23. }

    这里我们复制两个节点一个是我们的首节点prev当作我们的前驱节点,还有一个就是cur是我们的第二个节点也就是我们当前搜索的节点。

    如果我们的cur中的值为key那么我们的前驱节点直接指向我们cur的下个节点的地址,如果不是key,则我们的前驱节点等于我们当前的cur,cur接着向下指。最后我们再判断一下我们首节点的特殊情况,如果我们的首节点就是key的话,那么我们的首节点就是我们首节点指向的下一个节点。

    得到单链表的长度

    1. //得到单链表的长度
    2. public int size() {
    3. int size = 0;
    4. ListNode cur = this.head;
    5. while(cur!=null) {
    6. size++;
    7. cur = cur.next;
    8. }
    9. return size;
    10. }

    清空链表

    1. //清空链表
    2. public void clear() {
    3. this.head=null;
    4. }

  • 相关阅读:
    探索计算机的I/O控制方式:了解DMA控制器的作用与优势
    SpringBoot -集成Druid
    PostgreSQL插件的安装使用与删除
    分享一套好用的功能测试用例编写框架
    Linux通过PID号找到对应的进程名及所在目录
    leetcode-23.合并K个升序链表
    1155掷骰子等于目标和的方法数 (dfs + 记忆化搜索)
    hbuiderx基于安卓app运动员体能综合分析训练系统 微信小程序
    麦芽糖-刀豆球蛋白A,maltose-ConcanavalinA,刀豆球蛋白A-PEG-麦芽糖
    语义化标签
  • 原文地址:https://blog.csdn.net/m0_56911284/article/details/126600825