当前位置:   article > 正文

【数据结构】ArrayList与顺序表_顺序表和arrylist

顺序表和arrylist

本篇学习【数据结构】ArrayList与顺序表。先理解概念,再学会使用ArrayList并模拟实现。

目录

 线性表

顺序表

List介绍

ArrayList

1、ArrayList的介绍

 2、ArrayList的使用

2、1ArrayList的构造方法

 2、2ArrayList常见操作

3、ArrayList的遍历

 4、ArrayList模拟实现


 线性表

线性表(linear list)是n个具有相同特性的数据元素的有限序列。 线性表是一种在实际中广泛使用的数据结构,常见的线性表:顺序表、链表、栈、队列...

线性表在逻辑上是线性结构,也就说是连续的一条直线。但是在物理结构上并不一定是连续的,线性表在物理上存储时,通常以数组和链式结构的形式存储。

顺序表

顺序表是用一段物理地址连续的存储单元依次存储数据元素的线性结构,一般情况下采用数组存储。在数组上完成数据的增删查改。

List介绍

在集合框架中,List是一个接口,继承自Collection

Collection是一个接口,该接口中规范了后序容器中常用的一些方法。

Iterable也是一个接口,表示实现该接口的类是可以逐个元素进行遍历的

站在数据结构的角度来看,List就是一个线性表,即n个具有相同类型元素的有限序列,在该序列上可以执行增删 改查以及变量等操作

List的方法非常多。

注意:List是个接口,并不能直接用来实例化

ArrayList

1、ArrayList的介绍

在集合框架中,ArrayList是一个普通的类,实现了List接口,具体框架图如下:

说明

1. ArrayList是以泛型方式实现的,使用时必须要先实例化

2. ArrayList实现了RandomAccess接口,表明ArrayList支持随机访问

3. ArrayList实现了Cloneable接口,表明ArrayList是可以clone的

4. ArrayList实现了Serializable接口,表明ArrayList是支持序列化的

5. 和Vector不同,ArrayList不是线程安全的,在单线程下可以使用,在多线程中可以选择Vector或者 CopyOnWriteArrayList

6. ArrayList底层是一段连续的空间,并且可以动态扩容,是一个动态类型的顺序表

 2、ArrayList的使用

2、1ArrayList的构造方法

方法
ArrayList ( )
ArrayList ( Collection< ? extends E >c)
ArrayList ( int innitialCapacity)

解释:

方法1,无参构造

方法2,c为参数,实现了 Collection接口 。?为通配符,? extends E 为通配符上界,? 要么是E要么是E的子类

方法3,指定顺序表初始容量

public static void main(String[] args) {

// ArrayList创建,推荐写法

// 构造一个空的列表,对应方法1

List<Integer> list1 = new ArrayList<>();

// 构造一个具有10个容量的列表,对应方法3

List<Integer> list2 = new ArrayList<>(10);

list2.add(1);

list2.add(2);

list2.add(3);

//List<Integer>已经限定了,list2中只能存储整形元素

//对应方法2,list3包含list2的元素

ArrayList<Integer> list3 = new ArrayList<>(list2);

}

 2、2ArrayList常见操作

ArrayList虽然提供的方法比较多,但是常用方法如下所示

实例:

  1. public class Test {
  2. public static void main(String[] args) {
  3. List<String> list = new ArrayList<>();
  4. list.add("JavaSE");
  5. list.add("JavaWeb");
  6. list.add("JavaEE");
  7. list.add("JVM");
  8. list.add("测试课程");
  9. System.out.println(list);
  10. // 获取list中有效元素个数
  11. System.out.println(list.size());
  12. // 获取和设置index位置上的元素,注意index必须介于[0, size)间
  13. System.out.println(list.get(1));
  14. list.set(1, "JavaWEB");
  15. System.out.println(list.get(1));
  16. // 在list的index位置插入指定元素,index及后续的元素统一往后搬移一个位置
  17. list.add(1, "Java数据结构");
  18. System.out.println(list);
  19. // 删除指定元素,找到了就删除,该元素之后的元素统一往前搬移一个位置
  20. list.remove("JVM");
  21. System.out.println(list);
  22. // 删除list中index位置上的元素,注意index不要超过list中有效元素个数,否则会抛出下标越界异常
  23. list.remove(list.size() - 1);
  24. System.out.println(list);
  25. // 检测list中是否包含指定元素,包含返回true,否则返回false
  26. if(list.contains("测试课程")){
  27. list.add("测试课程");
  28. }
  29. // 查找指定元素第一次出现的位置:indexOf从前往后找,lastIndexOf从后往前找
  30. list.add("JavaSE");
  31. System.out.println(list.indexOf("JavaSE"));//0
  32. System.out.println(list.lastIndexOf("JavaSE"));//4
  33. // 使用list中[0, 4)之间的元素构成一个新的SubList返回,但是和ArrayList共用一个elementData数组
  34. List<String> ret = list.subList(0, 4);
  35. System.out.println(ret);//打印 [JavaSE, Java数据结构, JavaWEB, JavaEE]
  36. list.clear();//清空
  37. System.out.println(list.size());
  38. }
  39. }

 【注意】方法subList  使用list中[0, 4)之间的元素构成一个新的SubList返回,但是和ArrayList共用一个elementData数组

System.out.println(list);//[JavaSE, Java数据结构, JavaWEB, JavaEE]

List<String> ret = list.subList(0, 2);
System.out.println(ret);//打印 [JavaSE, Java数据结构]

ret.set(0,"测试课程");
System.out.println(list);//[测试课程, Java数据结构, JavaWEB, JavaEE]
System.out.println(ret);//[测试课程, Java数据结构] 

3、ArrayList的遍历

ArrayList 可以使用三方方式遍历:for循环+下标、foreach、使用迭代器

  1. public static void main(String[] args) {
  2. List<Integer> list = new ArrayList<>();
  3. list.add(1);
  4. list.add(2);
  5. list.add(3);
  6. list.add(4);
  7. list.add(5);
  8. // 使用下标+for遍历
  9. for (int i = 0; i < list.size(); i++) {
  10. System.out.print(list.get(i) + " ");
  11. }
  12. System.out.println();
  13. // 借助foreach遍历
  14. for (Integer integer : list) {
  15. System.out.print(integer + " ");
  16. }
  17. System.out.println();
  18. Iterator<Integer> it = list.listIterator();
  19. while(it.hasNext()){
  20. System.out.print(it.next() + " ");
  21. }
  22. System.out.println();
  23. }

注意:

1. ArrayList最长使用的遍历方式是:for循环+下标 以及 foreach

2. 迭代器是设计模式的一种,后序容器接触多了再给大家铺垫

 4、ArrayList模拟实现

结构:

异常

定义两个异常,顺序表为空异常及数组下表不合法

 顺序表为空异常

  1. public class EmptyException extends RuntimeException{
  2. public EmptyException() {
  3. super();
  4. }
  5. public EmptyException(String message) {
  6. super(message);
  7. }
  8. }

数组下表不合法

  1. public class PosIllegal extends RuntimeException{
  2. public PosIllegal(){
  3. }
  4. public PosIllegal(String message) {
  5. super(message);
  6. }
  7. }

MyArrayList类

共包含增删改查等方法。代码有注释

  1. public class MyArrayList {
  2. public int[] elem;
  3. public int usedSize;//0
  4. //默认容量
  5. private static final int DEFAULT_SIZE = 10;
  6. public MyArrayList() {
  7. this.elem = new int[DEFAULT_SIZE];
  8. }
  9. /**
  10. * 打印顺序表:
  11. * 根据usedSize判断即可
  12. */
  13. public void display() {
  14. for (int i = 0; i <usedSize ; i++) {
  15. System.out.print(elem[i]+" ");
  16. }
  17. System.out.println("");
  18. }
  19. // 新增元素,默认在数组最后新增
  20. public void add(int data) {
  21. if(isFull()){
  22. grow();
  23. }
  24. this.elem[this.usedSize++]=data;
  25. }
  26. /**
  27. * 判断当前的顺序表是不是满的!
  28. * @return true:满 false代表空
  29. */
  30. public boolean isFull() {
  31. return this.usedSize==elem.length;
  32. }
  33. //检查pos位置是否合法,用于add方法
  34. private boolean checkPosInAdd(int pos) throws PosIllegal{
  35. if(pos<0||pos>usedSize) {
  36. throw new PosIllegal("pos位置不合法");
  37. //合法
  38. }
  39. return false;
  40. }
  41. //检查pos位置是否合法,用于删,改,查
  42. private boolean checkPos(int pos) throws PosIllegal{
  43. if(pos<0||pos>=usedSize) {
  44. throw new PosIllegal("pos位置不合法");
  45. //合法
  46. }
  47. return false;
  48. }
  49. // 在 pos 位置新增元素
  50. public void add(int pos, int data) {
  51. try {
  52. checkPosInAdd(pos);
  53. if(isFull())
  54. grow();
  55. for (int i =usedSize-1 ; i>=pos ; i--) {
  56. elem[i+1]=elem[i];
  57. }
  58. elem[pos]=data;
  59. usedSize++;
  60. }catch(PosIllegal e){
  61. e.printStackTrace();
  62. System.out.println("插入元素pos位置不合法");
  63. }
  64. }
  65. //扩容
  66. private void grow(){
  67. this.elem=Arrays.copyOf(this.elem,2*this.elem.length);
  68. }
  69. // 判定是否包含某个元素
  70. public boolean contains(int toFind) {
  71. for (int i = 0; i < usedSize; i++) {
  72. if(elem[i]==toFind)
  73. return true;
  74. }
  75. return false;
  76. }
  77. // 查找某个元素对应的位置
  78. public int indexOf(int toFind) {
  79. for (int i = 0; i < usedSize; i++) {
  80. if(elem[i]==toFind)
  81. return i;
  82. }
  83. return -1;
  84. }
  85. // 获取 pos 位置的元素
  86. public int get(int pos) {
  87. try{
  88. checkEmpty();
  89. checkPos(pos);
  90. return elem[pos];
  91. }catch(PosIllegal e){
  92. e.printStackTrace();
  93. System.out.println("pos位置不合法");
  94. }catch(EmptyException e){
  95. e.printStackTrace();
  96. System.out.println("pos位置为空");
  97. }
  98. return -1;
  99. }
  100. //判断是否为空,抛出异常
  101. private void checkEmpty(){
  102. if(isEmpty())
  103. throw new EmptyException("顺序表为空");
  104. }
  105. //判断是否为空
  106. private boolean isEmpty() {
  107. return usedSize==0;
  108. }
  109. // 给 pos 位置的元素设为【更新为】 value
  110. public void set(int pos, int value) {
  111. try{
  112. checkEmpty();
  113. checkPos(pos);
  114. elem[pos]=value;
  115. }catch(PosIllegal e){
  116. e.printStackTrace();
  117. System.out.println("pos位置不合法");
  118. }catch(EmptyException e){
  119. e.printStackTrace();
  120. System.out.println("pos位置为空");
  121. }
  122. }
  123. /**
  124. * 删除第一次出现的关键字key
  125. * @param key
  126. */
  127. public void remove(int key) {
  128. try{
  129. checkEmpty();
  130. int pos = indexOf(key);
  131. for (int i = pos; i < usedSize-1; i++) {
  132. elem[i]=elem[i+1];
  133. }
  134. usedSize--;
  135. }catch (EmptyException e){
  136. e.printStackTrace();
  137. }
  138. }
  139. // 获取顺序表长度
  140. public int size() {
  141. return usedSize;
  142. }
  143. // 清空顺序表
  144. public void clear() {
  145. usedSize=0;
  146. }
  147. }

 测试类

  1. public class Test {
  2. public static void main(String[] args) {
  3. MyArrayList myArrayList=new MyArrayList();
  4. myArrayList.add(5);
  5. myArrayList.add(1);
  6. myArrayList.add(3);
  7. myArrayList.add(7);
  8. myArrayList.add(0);
  9. //打印数字为4的下标
  10. int ret= myArrayList.indexOf(4);
  11. System.out.println(ret);
  12. //移除数字3
  13. myArrayList.remove(3);
  14. //设置数组下标4的位置为1
  15. myArrayList.set(4,1);
  16. //打印数组
  17. myArrayList.display();
  18. }
  19. }

本篇【数据结构】ArrayList与顺序表结束

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

闽ICP备14008679号