快邀请你的冤种朋友一起来学习顺序表的底层逻辑:ArrayList集合

简介: 快邀请你的冤种朋友一起来学习顺序表的底层逻辑:ArrayList集合



一.ArrayList是什么

(1)为什么说ArrayList是一个动态顺序表

  • ArrayList底层是一段连续的空间,并且可以动态扩容,所以是一个动态的顺序表
  • 我们知道,我们实现顺序表的话,需要自己写它增删查改的方法,然后再调用;而ArrayList可以看作是一个已经实现好的顺序表,我们可以直接调用和使用

(2)从实现方面理解ArrayList

  • ArrayList是一个类,并且是一个泛型类(限制类型的作用),类中有许许多多的方法(可以类比String类)
  • 泛型类,所以使用的时候必须实例化
ArrayList<Integer> arrayList = new ArrayList<>();//实例化操作

(3)ArrayList实现的接口及拥有的功能

  • ArryList实现了List接口,List是一个线性表,所以ArrayList就具备了顺序表的功能(本节重点)

  • ArrayList实现了RandomAccess接口,表明ArrayList支持随机访问
  • ArrayList实现了Cloneable接口,表明ArrayList是可以clone的
  • ArrayList实现了Serializable接口,表明ArrayList是支持序列化的

(4)简单使用ArrayList

  • 常规的实例化

格式:ArrayList<类型>  引用  =  new  ArrayList<>()

ArrayList<Integer> arrayList = new ArrayList<>();//实例化操作
ArrayList<String> arrayList1 = new ArrayList<>();

  • 使用向上转型实例化

格式:List<类型>  引用  =  new  ArrayList<>()

List<Integer> list = new ArrayList<>();
List<String> list2 = new ArrayList<>();

  • 简单的添加数据和打印
public static void main(String[] args) {
        List<Integer> list = new ArrayList<>();
        list.add(1);
        list.add(3);
        list.add(1);
        list.add(4);
        list.add(5);
        list.add(2);
        list.add(0);
        System.out.println(list);
    }

二.ArrayList的构造方法与扩容机制

构造方法:

ArrayList()  无参构造
ArrayList(int initialCapacity)  指定顺序表初始容量
ArrayList(Collection<? extends E> c) 利用其他 Collection 构建 ArrayList

ArrayList的成员变量:

private static final int DEFAULT_CAPACITY = 10;
private static final Object[] EMPTY_ELEMENTDATA = {};
private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {};
transient Object[] elementData;
private int size;//记录当前顺序的大小

1.无参构造方法与扩容机制

(1)认识和使用无参构造方法

  • 无参的构造方法是这样实例化对象的
ArrayList<Integer> arrayList = new ArrayList<>();
  • 原码的无参构造方法
public ArrayList() {
        this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA;
    }

(2)如何执行无参构造方法

从上面我们可以得到一个结论:

结论一:使用无参构造方法实例化对象时,是没有给顺序表分配内存空间的

(3)扩容机制

  • 先看一段代码
public static void main(String[] args) {
        ArrayList<Integer> arrayList1 = new ArrayList<>();
        arrayList1.add(1);
        arrayList1.add(2);
        System.out.println(arrayList1);
    }

提出一个疑问?明明没有对该顺序表分配内存空间,为什么还可以对其赋值呢?下面走进add的原码一看究竟

  • 查看add的工作原理及增容机制

由上面我们可以得出结论:

结论二:第一次add(第一次增容)时,默认分配10的内存空间

  • 进行扩容操作的适时候

结论三:当扩容时,会按原内存的1.5倍进行扩容

2.带参数的构造方法

(1)认识和使用带参构造方法

  • 指定容量是是这样实例化对象的
ArrayList<Integer> arrayList = new ArrayList<>(15);//指定开辟15个空间
  • 查看带参数的构造方法原码
public ArrayList(int initialCapacity) {
        if (initialCapacity > 0) {
            this.elementData = new Object[initialCapacity];
        } else if (initialCapacity == 0) {
            this.elementData = EMPTY_ELEMENTDATA;
        } else {
            throw new IllegalArgumentException("Illegal Capacity: "+
                                               initialCapacity);
        }
    }

(2)如何执行构造方法

  • 当参数为负数时:运行时抛异常

3.利用其他 Collection 构建 ArrayList

这种构造方法是什么意思呢?也就是将其他的顺序表作为参数,下面一起来看看吧

(1)认识和使用该种构造方法

  • 使用该方法实例化对象
public static void main(String[] args) {
        ArrayList<Integer> arrayList = new ArrayList<>();
        arrayList.add(520);
        arrayList.add(1314);
        ArrayList<Integer> arrayList1 = new ArrayList<>(arrayList);//第三种构造方式
        System.out.println(arrayList1);
    }

  • 认识该构造方法的原码
public ArrayList(Collection<? extends E> c) {
        Object[] a = c.toArray();
        if ((size = a.length) != 0) {
            if (c.getClass() == ArrayList.class) {
                elementData = a;
            } else {
                elementData = Arrays.copyOf(a, size, Object[].class);
            }
        } else {
            // replace with empty array.
            elementData = EMPTY_ELEMENTDATA;
        }
    }

(2)认识原码的工作原理

  • 原码

  • 参数需要满足的条件

(3)使用该构造方法及错误原因

  • 通用的
public static void main(String[] args) {
        List<Integer> list = new ArrayList<>();
        List<Integer> list2 = new ArrayList<>(list);
        ArrayList<Integer> list3 = new ArrayList<>(list2);
        List<Integer> list4 = new ArrayList<>(list3);
        ArrayList<Number>  list5 = new ArrayList<>(list4);
    }

  • 错误案例

三.ArrayList常用方法介绍

常用的方法表格,如下所示:

方法 功能
boolean add(E e)  尾插 e
void add(int index, E element)  将 e 插入到 index 位置
boolean addAll(Collection<? extends E> c)  尾插 c 中的元素
E remove(int index)  删除 index 位置元素并返回该元素
boolean remove(Object o)  删除遇到的第一个 o
E get(int index)  获取下标 index 位置元素
E set(int index, E element)  将下标 index 位置元素设置为 element
void clear()  清空顺序表
boolean contains(Object o)  判断 o 是否在线性表中
int indexOf(Object o)  返回第一个 o 所在下标
int lastIndexOf(Object o)  返回最后一个 o 的下标
List<E> subList(int fromIndex, int toIndex)  截取部分 list

1.插入数据方法

(1)boolean add(E e)

默认尾插的方法

public static void main(String[] args) {
      ArrayList<Integer> arrayList = new ArrayList<>();
      arrayList.add(1);
        arrayList.add(2);
        arrayList.add(3);
        System.out.println(arrayList);
    }

(2)void add(int index, E element)

指定位置插入数据

public static void main(String[] args) {
      ArrayList<Integer> arrayList = new ArrayList<>();
      arrayList.add(0,9);
        arrayList.add(0,9);
        arrayList.add(0,7);
        System.out.println(arrayList);
    }

2.boolean contains(Object o) 方法

(1)方法目的:判断 o 是否在线性表中(o是一个字符串);也就是删除线性表中的存在字符串o(字符串o中的每一个字符都不能留)

(2)方法举例

  • str1 = "welcome to world",str2 = "come",要求,删除str1中存在的str2字符,并返回一个线性表
  • 代码:
private static ArrayList<Character> func(String str1,String str2) {
        //删除str1中的str2
        ArrayList<Character> arrayList = new ArrayList<>();
        for (int i = 0; i < str1.length(); i++) {
            char ch = str1.charAt(i);
            if(!str2.contains(ch+"")) {
                arrayList.add(ch);
            }
        }
        return arrayList;
    }
    public static void main(String[] args) {
        ArrayList<String> arrayList = new ArrayList<>();
        String str1 = "welcome to world";
        String str2 = "come";
        ArrayList<Character> arrayList1 = func(str1,str2);
        System.out.println(arrayList1);
    }

  • 本题细节点:字符+""(空字符串)可以变成字符串类型

3.List<E> subList(int fromIndex, int toIndex) 方法

(1)功能:截取指定范围内的数据

返回值:List接口定义的变量

(2)简单举例:

public static void main(String[] args) {
        ArrayList<Integer> arrayList = new ArrayList<>();
        arrayList.add(10);
        arrayList.add(20);
        arrayList.add(30);
        arrayList.add(40);
        arrayList.add(50);
        System.out.println(arrayList);
        List<Integer> list = arrayList.subList(1,3);
        System.out.println(list);
    }

(3)subList的特点

  • 先看一段代码
public static void main(String[] args) {
        ArrayList<Integer> arrayList = new ArrayList<>();
        arrayList.add(10);
        arrayList.add(20);
        arrayList.add(30);
        arrayList.add(40);
        System.out.println("修改前:");
        System.out.println("arrayList:"+arrayList);
        List<Integer> list = arrayList.subList(1,3);
        System.out.println("list:"+list);
        list.set(0,999);
        list.set(0,888);
        System.out.println("修改后:");
        System.out.println("arrayList:"+arrayList);
        System.out.println("list:"+list);
    }

为什么对截取后的空间进行数据的修改?原线性表的内容也会发生同样的改变?

  • 原因解释

截取后的变量,依然指向同一块空间;当修改的时候,修改的依然是原数组的内容

4.remove方法

(1)作用:删除线性表中的一个元素,并返回它的值

四.ArrayList的遍历方式

在ArrayListd的方法中,没有可以直接用来遍历的方法,但是我们可以通过自己编写的代码来遍历,如for循环等等,下面一一介绍

  • 最常规的方式,直接打印
public static  void main(String[] args) {
        ArrayList<Integer> arrayList = new ArrayList<>();
        arrayList.add(5);
        arrayList.add(2);
        arrayList.add(0);
        System.out.println(arrayList);
}

1.for循环遍历

(1)常规for循环

求得该线性表的元素个数即可

public static void main(String[] args) {
        ArrayList<Integer> arrayList = new ArrayList<>();
        arrayList.add(5);
        arrayList.add(2);
        arrayList.add(0);
        for (int i = 0; i < arrayList.size(); i++) {
            System.out.print(arrayList.get(i)+" ");
        }
    }

(2)加强for循环

for循环中也可可以这么写:int x : arrayList(那么里面的数据类就是发生了拆箱的操作)

public static void main(String[] args) {
        ArrayList<Integer> arrayList = new ArrayList<>();
        arrayList.add(5);
        arrayList.add(2);
        arrayList.add(0);
        for (Integer x:arrayList) {
            System.out.print(x+" ");
        }
    }

2.使用迭代器遍历

可以使用迭代器遍历的前提是该集合实现了Iterable,载使用前,需要先获取线性表的迭代器

(1)从前往后迭代:Iterator接口

public static void main(String[] args) {
        ArrayList<Integer> arrayList = new ArrayList<>();
        arrayList.add(5);
        arrayList.add(2);
        arrayList.add(0);
        Iterator<Integer> it = arrayList.iterator();//获取到线性表的迭代器
        //开始遍历
        while(it.hasNext()) {
            System.out.print(it.next()+" ");//每次打印它的下一个
        }
    }

图解:

(2)从前往后遍历:ListIterator接口

public static  void main(String[] args) {
        ArrayList<Integer> arrayList = new ArrayList<>();
        arrayList.add(5);
        arrayList.add(2);
        arrayList.add(0);
        ListIterator<Integer> it2 = arrayList.listIterator();
        while(it2.hasNext()) {
            System.out.print(it2.next()+" ");
        }
    }

与Iterator接口的关系:继承关系

(3)从后往前遍历:ListIterator接口(加参数即可)

public static  void main(String[] args) {
        ArrayList<Integer> arrayList = new ArrayList<>();
        arrayList.add(5);
        arrayList.add(2);
        arrayList.add(0);
        ListIterator it3 = arrayList.listIterator(arrayList.size());//让迭代器指向最后一个元素后面
        while (it3.hasPrevious()) {
            System.out.print(it3.previous()+" ");//打印前一个
        }
    }


本次的介绍就到这里了,小伙伴们快去试试吧

相关文章
|
存储 Java
ArrayList的初始化容量与扩容机制解析
ArrayList的初始化容量与扩容机制解析
|
存储 算法 Java
HashMap 之底层数据结构和扩容机制
HashMap 之底层数据结构和扩容机制
1962 1
|
安全 Java API
ArrayList 全面详解
关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。本文详细解析了Java集合框架中的ArrayList,包括其定义、特点、成员变量、构造函数、API、主要方法和扩容机制等。欢迎留言交流。
|
9月前
|
算法 安全 Java
2025 校招必看:Java 开发面试核心知识点深度解析及最新笔面试题汇总
本文针对2025校招Java开发面试,系统梳理了Java基础、集合框架、多线程并发、JVM等核心知识点,并附带最新笔面试题。内容涵盖封装、继承、多态、异常处理、集合类使用、线程同步机制、JVM内存模型及垃圾回收算法等。同时深入探讨Spring、数据库(MySQL索引优化、Redis持久化)、分布式系统(CAP理论、分布式事务)等相关知识。通过理论结合实例解析,帮助考生全面掌握面试要点,提升实战能力,为成功拿下Offer奠定坚实基础。
913 3
|
存储 安全 Java
面试高频:Synchronized 原理,建议收藏备用 !
本文详解Synchronized原理,包括其作用、使用方式、底层实现及锁升级机制。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
面试高频:Synchronized 原理,建议收藏备用 !
|
安全 Java
Java“不兼容类型” 错误怎么查找解决
在 Java 中遇到“不兼容类型”错误时,首先理解错误信息,它表明试图将一种类型赋给不兼容的类型。检查代码中类型不匹配的赋值、方法调用参数类型不匹配、表达式类型不兼容及泛型类型不匹配等问题。解决方法包括进行类型转换、修改代码逻辑、检查方法参数和返回类型以及处理泛型类型不匹配。通过这些步骤,可以有效解决“不兼容类型”错误,确保代码类型兼容性良好。
2911 9
|
存储 监控 数据库
什么是聚集索引和非聚集索引?
【8月更文挑战第3天】
9527 6
|
存储 Java API
Java数据结构之ArrayList(如果想知道Java中有关ArrayList的知识点,那么只看这一篇就足够了!)
Java数据结构之ArrayList(如果想知道Java中有关ArrayList的知识点,那么只看这一篇就足够了!)
Java数据结构之ArrayList(如果想知道Java中有关ArrayList的知识点,那么只看这一篇就足够了!)
|
Java
认真研究Java集合之ArrayList的实现原理
认真研究Java集合之ArrayList的实现原理
188 0
|
Java Maven Windows
java -jar 启动 boot 程序 no main manifest attribute, in .\vipsoft-model-web-0.0.1-SNAPSHOT.jar
java -jar 启动 boot 程序 no main manifest attribute, in .\vipsoft-model-web-0.0.1-SNAPSHOT.jar
290 0

热门文章

最新文章