数据结构(Java实现):ArrayList

发布于:2024-05-17 ⋅ 阅读:(187) ⋅ 点赞:(0)

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构造方法分析

ArrayList有三个构造方法
在这里插入图片描述
1.无参的构造方法
在这里插入图片描述
首先我们看到一个elementData为ArrayList定义好的一个数组,默认为空,指数一个数组引用。然后看到DEFAULTCAPACITY_EMPTY_ELEMENTDATA
在这里插入图片描述
可以看到并没有给数组分配内存。

2.带一个int参数的构造方法(可以给ArrayList设置一个初始容量)
在这里插入图片描述
先看到if判断语句,当传入参数大于0时,直接分配一个参数大小的空间
在这里插入图片描述
传入参数等于0时与无参构造方法一样,并不会给数组分配空间
在这里插入图片描述
传入参数小于0就会抛出一个异常
在这里插入图片描述
3.带一个类型变量参数的构造方法
在这里插入图片描述
我们先对传入的参数进行解析Collection<? extends E> c
在这里插入图片描述
这样我们就可以传入一个ArrayList变量,举例如下

public static void main(String[] args) {
        ArrayList<Integer> arrayList=new ArrayList<>();
        arrayList.add(10);
        arrayList.add(20);
        
        ArrayList<Integer> arrayList1=new ArrayList<>(arrayList);
        /*传入一个ArrayList变量,这个参数需要满足了下几个条件才能够传递
        *1.ArrayList实现了Collection接口
        *2.参数类型是arrayList1指定的泛型本身“Integer”
        */
    }

3.ArrayList的add方法以及扩容机制

在这里插入图片描述
看到上面的add方法,左边的add方法通过调用右边的add方法进行数据的插入操作,下面我们重点来讲解左边这个add方法
在这里插入图片描述
首先看到if判断语句,当数据的数量等于数组的长度就会调用grow方法进行扩容。
在这里插入图片描述
grow方法又调用了另一个grow方法
在这里插入图片描述
先看到if里面的判断语句,oldCapacity > 0 || elementData != DEFAULTCAPACITY_EMPTY_ELEMENTDATA当数组容量初始为DEFAULTCAPACITY_EMPTY_ELEMENTDATA(空)的时候进入else语句elementData = new Object[Math.max(DEFAULT_CAPACITY, minCapacity)];给数组分配空间为DEFAULT_CAPACITYminCapacity的最大值
在这里插入图片描述
看到DEFAULT_CAPACITY的值为10,所以当数组初始空间为空时就会给数组分配一个容量为10的内存。正常进入if语句就会触发ArrayList的扩容机制
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
扩容的空间大概是原来的1.5倍,通过一步一步调用上面的方法(了解)

4.ArrayList常用方法

在这里插入图片描述
举例如下

public static void main(String[] args) {
List<String> list = new ArrayList<>();
list.add("JavaSE");
list.add("JavaWeb");
list.add("JavaEE");
list.add("JVM");
list.add("测试课程");
System.out.println(list);
// 获取list中有效元素个数
System.out.println(list.size());
// 获取和设置index位置上的元素,注意index必须介于[0, size)间
System.out.println(list.get(1));
list.set(1, "JavaWEB");
System.out.println(list.get(1));
// 在list的index位置插入指定元素,index及后续的元素统一往后搬移一个位置
list.add(1, "Java数据结构");
System.out.println(list);
// 删除指定元素,找到了就删除,该元素之后的元素统一往前搬移一个位置
list.remove("JVM");
System.out.println(list);
// 删除list中index位置上的元素,注意index不要超过list中有效元素个数,否则会抛出下标越界异常
list.remove(list.size()-1);
System.out.println(list);
// 检测list中是否包含指定元素,包含返回true,否则返回false
if(list.contains("测试课程")){
list.add("测试课程");
}
// 查找指定元素第一次出现的位置:indexOf从前往后找,lastIndexOf从后往前找
list.add("JavaSE");
System.out.println(list.indexOf("JavaSE"));
System.out.println(list.lastIndexOf("JavaSE"));
// 使用list中[0, 4)之间的元素构成一个新的SubList返回,但是和ArrayList共用一个elementData数组
List<String> ret = list.subList(0, 4);
System.out.println(ret);
list.clear();
System.out.println(list.size());
}

5.ArrayList遍历

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

public static void main(String[] args) {
List<Integer> list = new ArrayList<>();
list.add(1);
list.add(2);
list.add(3);
list.add(4);
list.add(5);
// 使用下标+for遍历
for (int i = 0; i < list.size(); i++) {
System.out.print(list.get(i) + " ");
}
System.out.println();
// 借助foreach遍历
for (Integer integer : list) {
System.out.print(integer + " ");
}
System.out.println();
Iterator<Integer> it = list.listIterator();
while(it.hasNext()){
System.out.print(it.next() + " ");
}
System.out.println();
}

注意

  1. ArrayList最长使用的遍历方式是:for循环+下标 以及 foreach
  2. 迭代器是设计模式的一种,后序容器使用更多,在进行详细介绍

网站公告

今日签到

点亮在社区的每一天
去签到