ArrayList底层源码分析

ArrayList底层源码分析

1. ArrayList集合底层数据结构

  1. ArrayList集合介绍
  • List 接口的可调整大小的数组实现。

  • 数组:一旦初始化长度就不可以发生改变

  1. 数组结构介绍
  • 增删慢:每次删除元素,都需要更改数组长度、拷贝以及移动元素位置。
  • 查询快:由于数组在内存中是一块连续空间,因此可以根据地址+索引的方式快速获取对应位置上的元素。

2. ArrayList继承关系

2.1 Serializable标记性接口

  1. 介绍: 类的序列化由实现java.io.Serializable接口的类启用。 不实现此接口的类将不会使任何状态序列化或反

    序列化。 可序列化类的所有子类型都是可序列化的。 序列化接口没有方法或字段,仅用于标识可串行化的语

    义。

    • 序列化:将对象的数据写入到文件(写对象)

    • 反序列化:将文件中对象的数据读取出来(读对象)

  2. Serializable源码介绍

    public interface Serializable { }
    
  3. 序列化流序列化和反序列化集合

    //定义Student实体类
    package com.ljt.ArrayList.demo01;
    
    import java.io.Serializable;
    
    public class Student implements Serializable {
    
        private String name;
    
        private Integer age;
    
        public Student(String name, Integer age) {
            this.name = name;
            this.age = age;
        }
        public String getName() {
            return name;
        }
    
        public void setName(String name) {
            this.name = name;
        }
    
        public Integer getAge() {
            return age;
        }
    
        public void setAge(Integer age) {
            this.age = age;
        }
    
        @Override
        public String toString() {
            StringBuilder sb=new StringBuilder();
            sb.append("[").append("name=").append(this.name).
                    append(", ").append("age=").append(this.age)
                    .append("]");
          return sb.toString();
        }
    }
    

    序列化和反序列化

    package com.ljt.ArrayList.demo01;
    
    import java.io.*;
    import java.util.ArrayList;
    import java.util.List;
    
    public class StudentDemo {
        public static void main(String[] args) throws Exception {
            List<Student> list=new ArrayList<>();
            ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("D:\\IdeaProjects\\集合\\a.txt"));
            list.add(new Student("张三",15));
            list.add(new Student("李四",18));
            list.add(new Student("王五",16));
            oos.writeObject(list);
            oos.close();
            ObjectInputStream ois = new ObjectInputStream(new FileInputStream("D:\\IdeaProjects\\集合\\a.txt"));
            List<Student> students = (List<Student>)ois.readObject();
            System.out.println(students);
            ois.close();
        }
    }
    
    [[name=张三, age=15], [name=李四, age=18], [name=王五, age=16]]
    

2.2 Cloneable 标记性接口

  1. 介绍:一个类实现 Cloneable 接口来指示 Object.clone() 方法,该方法对于该类的实例进行字段的复制是合

    法的。在不实现 Cloneable 接口的实例上调用对象的克隆方法会导致异常 CloneNotSupportedException 被抛出。简言之:克隆就是依据已经有的数据,创造一份新的完全一样的数据拷贝

  2. Cloneable源码介绍

    public interface Cloneable { }
    
  3. 克隆的前提条件

    • 被克隆对象所在的类必须实现 Cloneable 接口
    • 必须重写 clone 方法
  4. clone的基本使用

    package com.ljt.ArrayList.demo01;
    
    import java.util.ArrayList;
    
    public class CloneDemo {
        public static void main(String[] args) {
            ArrayList<String> list = new ArrayList<>();
            list.add("生活就像海洋");
            list.add("只有意志坚强的人才能到达彼岸");
            Object clone = list.clone();
            System.out.println(list);
            System.out.println(clone);
            System.out.println(list == clone);//false
        }
    }
    
  5. clone源码分析

    public Object clone() {
            try {
                ArrayList<?> v = (ArrayList<?>) super.clone();
                v.elementData = Arrays.copyOf(elementData, size);
                v.modCount = 0;
                return v;
            } catch (CloneNotSupportedException e) {
                // this shouldn't happen, since we are Cloneable
                throw new InternalError(e);
            }
        }
    

    案例:已知 A 对象的姓名为豹子头林冲,年龄30 。由于项目特殊要求需要将该对象的数据复制另外一个对象B 中,并且此后 A 和 B 两个对象的数据不会相互影响

    案例:已知 A 对象的姓名为鲁智深,年龄30,技能为倒拔垂杨柳 (技能为一个引用数据类型 Skill ),由于项目特殊要求需要将该对象的数据复制另外一个对象 B 中,并且此后 A 和 B 两个对象的数据不会相互影响

方式一:创建两个对象模拟

  1. 准备学生类

    package com.ljt.ArrayList.demo01;
    
    import java.io.Serializable;
    
    public class Student implements Serializable,Cloneable {
    
        private String name;
    
        private Integer age;
    
        public Student(String name, Integer age) {
            this.name = name;
            this.age = age;
        }
        
        public Student(){
            
        }
        
        public String getName() {
            return name;
        }
    
        public void setName(String name) {
            this.name = name;
        }
    
        public Integer getAge() {
            return age;
        }
    
        public void setAge(Integer age) {
            this.age = age;
        }
    
        @Override
        public String toString() {
            StringBuilder sb=new StringBuilder();
            return sb.append("[").append("name=").append(this.name).append(", ")
                    .append("age=").append(this.age).append("]").toString();
        }
    }
    
  2. 准备测试类

    package com.ljt.ArrayList.demo01;
    
    public class Test01 {
        public static void main(String[] args) throws CloneNotSupportedException {
            //创建学生对象
            Student stu1 = new Student("豹子头林冲", 29);
            //再次创建一个新的学生对象
            Student stu2 = new Student();
            //把stu1对象name的值取出来赋值给stu2对象的name 
            stu2.setName(stu1.getName());
            //把stu1对象age的值取出来赋值给stu2对象的age 
            stu2.setAge(stu1.getAge());
            System.out.println(stu1 == stu2);
            System.out.println(stu1);
            System.out.println(stu2);
            System.out.println("----此时不管修改哪个对象的内容,stu1和stu2都不会受到影响----");
            stu1.setName("扈三娘");
            System.out.println(stu1);
            System.out.println(stu2);
        }
    }
    
    false
    [name=豹子头林冲, age=29]
    [name=豹子头林冲, age=29]
    ----此时不管修改哪个对象的内容,stu1和stu2都不会受到影响----
    [name=扈三娘, age=29]
    [name=豹子头林冲, age=29]
    

方式二:使用克隆

  • 浅拷贝

    1.定义Javabean类

    //学生的技能类
    public class Skill implements Cloneable {
        private String skillName;
    
        public Skill() {
        }
    
        public Skill(String skillName) {
            this.skillName = skillName;
        }
    
        public String getSkillName() {
            return skillName;
        }
    
        public void setSkillName(String skillName) {
            this.skillName = skillName;
        }
    
        @Override
        public String toString() {
            return "Skill{" + "skillName='" + skillName + '\'' + '}';
        }
    }
    

    Student

    public class Student implements Serializable,Cloneable {
    
        private String name;
    
        private Integer age;
    
        private Skill skill;
    
        public Student(){
    
        }
    
        public Student(String name, Integer age, Skill skill) {
            this.name = name;
            this.age = age;
            this.skill = skill;
        }
    
        public Skill getSkill() {
            return skill;
        }
    
        public void setSkill(Skill skill) {
            this.skill = skill;
        }
    
        public String getName() {
            return name;
        }
    
        public void setName(String name) {
            this.name = name;
        }
    
        public Integer getAge() {
            return age;
        }
    
        public void setAge(Integer age) {
            this.age = age;
        }
    
        @Override
        public String toString() {
            return "Student{" +
                    "name='" + name + '\'' +
                    ", age=" + age +
                    ", skill=" + skill +
                    '}';
        }
    
        @Override
        protected Student clone() throws CloneNotSupportedException {
            return (Student)super.clone();
        }
    }
    
  1. 定义测试类

    public class Test01 { 
        public static void main(String[] args) throws CloneNotSupportedException { 
            //用自定义对象演示 深浅拷贝 
            Skill skill = new Skill("倒拔垂杨柳"); 
            Student s = new Student("鲁智深",31,skill); 
            //调用clone方法进行克隆 
            Student obj = s.clone(); 
            //比较地址 
            System.out.println(s == obj); 
            System.out.println("被克隆对象: "+s); 
            System.out.println("克隆出来的对象: "+obj); 
            System.out.println("----华丽的分割线----");
            //克隆之后,更改skill中的数据 
            skill.setSkillName("荷花酒"); 
            //更改克隆后对象的数据 
            obj.setName("扈三娘"); 
            obj.setAge(19); 
            System.out.println("被克隆对象: "+s); 
            System.out.println("克隆出来的对象: "+obj); 
        } 
    }        
    控制台输出
    false
    被克隆对象: Student{name='鲁智深', age=31, skill=Skill{skillName='倒拔垂杨柳'}}
    克隆出来的对象: Student{name='鲁智深', age=31, skill=Skill{skillName='倒拔垂杨柳'}}
    ----华丽的分割线----
    被克隆对象: Student{name='鲁智深', age=31, skill=Skill{skillName='荷花酒'}}
    克隆出来的对象: Student{name='扈三娘', age=19, skill=Skill{skillName='荷花酒'}}    
    

    存在的问题: 基本数据类型可以达到完全复制,引用数据类型则不可以

    原因: 在学生对象s被克隆的时候,其属性skill(引用数据类型)仅仅是拷贝了一份引用,因此当skill的值发生改

    变时,被克隆对象s的属性skill也将跟随改变

  • 深拷贝
  1. 定义Javabean类

    package com.ljt.ArrayList.demo01;
    
    public class Skill implements Cloneable {
        private String skillName;
    
        public Skill() {
        }
    
        public Skill(String skillName) {
            this.skillName = skillName;
        }
    
        public String getSkillName() {
            return skillName;
        }
    
        public void setSkillName(String skillName) {
            this.skillName = skillName;
        }
        //重写clone方法,并把访问权限改为public
        @Override
        public Object clone() throws CloneNotSupportedException {
            return super.clone();
        }
    
        @Override
        public String toString() {
            return "Skill{" + "skillName='" + skillName + '\'' + '}';
        }
    }
    

    学生类

    public class Student implements Cloneable{ 
        //姓名
        private String name; 
        //年龄 
        private Integer age;
        //技能 
        private Skill skill; 
        public Student() { 
        }
        public Student(String name, Integer age, Skill skill) {
            this.name = name; 
            this.age = age; 
            this.skill = skill;
        }
        public String getName() { 
            return name; 
        }public void setName(String name) { 
            this.name = name;
        }
        public Integer getAge() {
            return age; 
        }
        public void setAge(Integer age) { 
            this.age = age;
        }
        public Skill getSkill() {
            return skill;
        }
        public void setSkill(Skill skill) { 
            this.skill = skill;
        }
        @Override
        public String toString() { 
            return "Student{" + "name='" + name + '\'' + ", age=" + age + ", skill=" + skill + '}'; 
        }
        //深拷贝 
        @Override
        public Student clone() throws CloneNotSupportedException { 
            //调用超类Object中方法clone进行对象克隆,得到一个新的学生对象
            Student newStu = (Student) super.clone(); 
            //调用学生类其属性skill的clone方法,对属性进行克隆 
            Skill s = (Skill)this.skill.clone(); 
            //再将克隆的Skill设置给克隆出来的学生对象 
            newStu.setSkill(s); 
            //返回克隆出来的学生对象 
            return newStu; 
        } 
    }
    
    控制台输出
    false
    被克隆对象: Student{name='鲁智深', age=31, skill=Skill{skillName='倒拔垂杨柳'}}
    克隆出来的对象: Student{name='鲁智深', age=31, skill=Skill{skillName='倒拔垂杨柳'}}
    ----华丽的分割线----
    被克隆对象: Student{name='鲁智深', age=31, skill=Skill{skillName='荷花酒'}}
    克隆出来的对象: Student{name='鲁智深', age=19, skill=Skill{skillName='倒拔垂杨柳'}}
    

2.3 RandomAccess标记接口

  1. 介绍 标记接口由 List 实现使用,以表明它们支持快速(通常为恒定时间)随机访问,此接口的主要目的是允许通用算法更改其行为,以便在应用于随机访问列表顺序访问列表时提供良好的性能。

    用于操纵随机访问列表的最佳算法(例如 ArrayList )可以在应用于顺序访问列表时产生二次行为(如

    LinkedList )。 鼓励通用列表算法在应用如果将其应用于顺序访问列表之前提供较差性能的算法时,检查

    给定列表是否为 instanceof ,并在必要时更改其行为以保证可接受的性能。

    人们认识到,随机访问和顺序访问之间的区别通常是模糊的。 例如,一些 List 实现提供渐近的线性访问时

    间,如果它们在实践中获得巨大但是恒定的访问时间。 这样的一个 List 实现应该通常实现这个接口。 根据

    经验, List 实现应实现此接口,如果对于类的典型实例,此循环:

    for (int i=0, n=list.size(); i < n; i++) list.get(i);
    

    比这个循环运行得更快:

    for (Iterator i=list.iterator(); i.hasNext(); ) i.next();
    
  2. 案例演示一

    public class RandomAccessDemo {
        public static void main(String[] args) {
            List<String> list = new ArrayList<>();
            //添加10W条数据
            for (int i = 0; i < 100000; i++) {
                list.add(i + "a");
            }
            System.out.println("----通过索引(随机访问:)----");
            long startTime = System.currentTimeMillis();
            for (int i = 0; i < list.size(); i++) {
                //仅仅为了演示取出数据的时间,因此不对取出的数据进行打印
                list.get(i);
            }
            long endTime = System.currentTimeMillis();
            System.out.println("随机访问: " + (endTime - startTime)+"ms");
            System.out.println("----通过迭代器(顺序访问:)----");
            startTime = System.currentTimeMillis();
            Iterator<String> it = list.iterator();
            while (it.hasNext()) {
                //仅仅为了演示取出数据的时间,因此不对取出的数据进行打印
                it.next();
            }
            endTime = System.currentTimeMillis();
            System.out.println("顺序访问: " + (endTime - startTime)+"ms");
        }
    }
    
    控制台输出
    ----通过索引(随机访问:)----
    随机访问: 1ms
    ----通过迭代器(顺序访问:)----
    顺序访问: 2ms
    
    
    
  3. 案例演示二

    public class Test02 { 
        public static void main(String[] args) { 
        //创建LinkedList集合 
        List<String> list = new LinkedList<>(); 
        //添加10W条数据
        for (int i = 0; i < 100000; i++) { 
            list.add(i+"a");
        }
        System.out.println("----通过索引(随机访问:)----"); 
        long startTime = System.currentTimeMillis(); 
        for (int i = 0; i < list.size(); i++) { 
            //仅仅为了演示取出数据的时间,因此不对取出的数据进行打印 
            list.get(i); 
        }
        long endTime = System.currentTimeMillis();
        System.out.println("随机访问: "+(endTime-startTime)+"ms"); 
        System.out.println("----通过迭代器(顺序访问:)----"); 
        startTime = System.currentTimeMillis(); 
        Iterator<String> it = list.iterator(); 
        while (it.hasNext()){ 
            //仅仅为了演示取出数据的时间,因此不对取出的数据进行打印 
            it.next(); 
        }
        endTime = System.currentTimeMillis(); 
        System.out.println("顺序访问: "+(endTime-startTime)+"ms");
       } 
    }
    
    控制台输出
    ----通过索引(随机访问:)----
    随机访问: 22131ms
    ----通过迭代器(顺序访问:)----
    顺序访问: 4ms
    

为什么LinkedList随机访问比顺序访问要慢这么多?

源码分析

随机访问

//每次LinkedList对象调用get方法获取元素,都会执行以下代码 
list.get(i); 
public class LinkedList<E> { 
    public E get(int index) {
        //检验是否有效 
        checkElementIndex(index); 
        //调用node(index) 
        return node(index).item; 
    }
    //node方法 
    Node<E> node(int index) { 
    //node方法每次被调用的时候都会根据集合size进行折半动作 
    //判断get方法中的index是小于集合长度的一半还是大于 
        if (index < (size >> 1)) { 
            //如果小于就从链表的头部一个个的往后找 
            Node<E> x = first;
            for (int i = 0; i < index; i++)
                x = x.next; 
            return x;
            } else {
            Node<E> x = last;
            //如果大于就从链表的尾部一个个的往前找 
            for (int i = size - 1; i > index; i--)
                x = x.prev;
            return x; 
        } 
    } 
}

顺序访问

//获取迭代器的时候,会执行以下代码 
Iterator<String> it = list.iterator(); 
//AbstractList为LinkedList父类的父类 
public abstract class AbstractList<E> {
    public ListIterator<E> listIterator() { 
        //返回一个列表迭代器,且指定参数为0 return listIterator(0); 
    } 
}
public class LinkedList<E>{ 
    public ListIterator<E> listIterator(int index) { 
        //检查索引位置
        checkPositionIndex(index); 
        //返回ListItr对象 
        return new ListItr(index);
    }
    //LinkedList迭代器实现类 
    private class ListItr implements ListIterator<E> {
        private Node<E> lastReturned; 
        private Node<E> next;
        private int nextIndex; 
        //将实际修改集合次数赋值给预期修改次数 
        private int expectedModCount = modCount;
       ListItr(int index) {
       //判断 0 == size,实际上就是调用 node(index)方法
          next = (index == size)? null : node(index);
        //将index的值赋值给 nextIndex,便于下次查找
      nextIndex = index;
   }
}
    Node<E> node(int index) { 
        //在获取迭代器的时候也会进行折半的动作 
        //但是在获取迭代器的时候 index 一定是0,因此 if的条件成立 
        if (index < (size >> 1)) {
            Node<E> x = first;
            //由于循环条件不成立,不会执行
            x.next;
            for (int i = 0; i < index; i++)
                x = x.next;
            return x; 
            //返回第一个元素 
        } else {
            Node<E> x = last;
            for (int i = size - 1; i > index; i--) 
                x = x.prev;
            return x;
        }
    }
}
//迭代器调用 hasNext()方法的时候,会执行以下代码
private class ListItr implements ListIterator<E> { 
    public boolean hasNext() { 
        //如果nextIndex < 集合的长度,就说明还有元素,可以进行next 
        return nextIndex < size; 
    }
}
//当迭代器调用it.next();方法的时候会执行以下代码 
it.next(); 
public class LinkedList<E>{ 
    private class ListItr implements ListIterator<E> { 
        public E next() { 
            checkForComodification();
            //检查集合实际修改次数和预期次数是否一样
            //再次判断是否有元素 
            if (!hasNext()) 
                throw new NoSuchElementException(); 
            //将链表的第一个元素赋值给lastReturned 
            lastReturned = next; 
            //将下一个元素赋值给next
            next = next.next;
            //nextIndex++ 
            nextIndex++;
            //返回第一个元素 
            return lastReturned.item;
        } 
    }
}

小结: 由于随机访问的时候源码底层每次都需要进行折半的动作,再经过判断是从头还是从尾部一个个寻找。

而顺序访问只会在获取迭代器的时候进行一次折半的动作,以后每次都是在上一次的基础上获取下一个元素。

因此顺序访问要比随机访问快得多

  1. 实际开发应用场景

    public class Test01 { 
        //创建JdbcTemplate对象 
        JdbcTemplate jt = new JdbcTemplate(JDBCUtils.getDataSource()); 
        //查询出基础班在读男学员的所有信息,且按成绩的降序输出到控制台上(利用JDBC)
        @Test public void fun2() throws Exception { 
            //拼写SQL 
            String sql = "select * from stutb where sex = ? and type like ? order by score desc"; 
            //调用方法查询 将结果集的每一行都封装成一个Stutb对象,并把每一个对象都添加到集合 
            List<Stutb> list = jt.query(sql, new BeanPropertyRowMapper<Stutb>(Stutb.class), "男", "%基础班%"); 
            //在遍历集合取出结果集之前面临一个问题,使用普通for遍历好 还是使用迭代器(增强for)? 
            //特别是数据量特别大的时候一定要考虑! 
            //对返回的集合进行判断,如果返回的集合实现了 RandomAccess 就使用 普通for 
            //否则使用迭代器(增强for) 
            if(list instanceof RandomAccess){ 
                for (int i = 0; i < list.size(); i++) { 
                    System.out.println(list.get(i)); 
                }
            }else {
                for (Stutb stutb : list) { 
                    System.out.println(stutb);
                }
            }
        } 
    }
    

3. ArrayList源码分析

3.1 构造方法

ConstructorConstructor描述
ArrayList()构造一个初始容量为十的空列表。
ArrayList(int initialCapacity)构造具有指定初始容量的空列表。
ArrayList(Collection<? extends E> c)构造一个包含指定集合的元素的列表,按照它们由集合的迭代器返回的顺序。

3.2 案例演示

案例一:

  1. 空参构造ArrayList()
public class Test01 {
    public static void main(String[] args) { 
        //这行代码做了什么?
        //真的构造一个初始容量为十的空列表? 
        ArrayList<String> list = new ArrayList<String>();
    } 
}
  1. 源码分析
public class ArrayList<E> { 
    //默认初始容量
    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;
    public ArrayList() {
        this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA;
    } 
}
  1. 结论

通过空参构造方法创建集合对象并未构造一个初始容量为十的空列表,仅仅将

DEFAULTCAPACITY_EMPTY_ELEMENTDATA的地址赋值给elementData

案例二:

  1. 指定容量ArrayList(int initialCapacity)
public class Test01 {
    public static void main(String[] args) { 
        //这行代码ArrayList底层做了什么? 
        ArrayList<String> list = new ArrayList<String>(5); 
    } 
}
  1. 源码分析
public class ArrayList<E> { 
    public ArrayList(int initialCapacity) { 
        //initialCapacity = 5
        //判断初始容量initialCapacity是否大于0 
        if (initialCapacity > 0) { 
            //创建一个数组,且指定长度为initialCapacity 
            this.elementData = new Object[initialCapacity]; 
        } else if (initialCapacity == 0) { 
            //如果initialCapacity容量为0,把EMPTY_ELEMENTDATA的地址赋值给elementData this.elementData = EMPTY_ELEMENTDATA;
        } else { 
            //以上两个条件都不满足报错 
            throw new IllegalArgumentException("Illegal Capacity: "+ initialCapacity); }     } 
}
  1. 结论 根据 ArrayList 构造方法参数创建指定长度的数组

案例三:

  1. ArrayList(Collection<? extends E> c)
public class Test01 { 
    public static void main(String[] args) {
        ArrayList<String> list = new ArrayList<String>(); 
        list.add("aaa");
        list.add("bbb");
        list.add("ccc"); 
        //这行代码做了什么? 
        ArrayList<String> list1 = new ArrayList<>(list);
        for (String s : list1) {
            System.out.println(s); 
           } 
       }
    }
  1. 源码分析
public ArrayList(Collection<? extends E> c) {
        //将构造方法中的参数转成数组
        elementData = c.toArray();
      // 将elementData的长度赋值给 集合长度size,且判断是否不等于 0
        if ((size = elementData.length) != 0) {
           // 判断elementData 和 Object[] 是否为不一样的类型
            if (elementData.getClass() != Object[].class)
                //如果不一样,使用Arrays的copyOf方法进行元素的拷贝
                elementData = Arrays.copyOf(elementData, size, Object[].class);
        } else {
            // 就把空数组的地址赋值给集合存元素的数组
            this.elementData = EMPTY_ELEMENTDATA;
        }
    }
    //将集合转数组的方法
    public Object[] toArray() {
        //调用数组工具类的方法
        return Arrays.copyOf(elementData, size);
    }
}
class Arrays {
    public static <T> T[] copyOf(T[] original, int newLength) {
        //再次调用方法得到一个数组
        return (T[]) copyOf(original, newLength, original.getClass());
    }
    public static <T,U> T[] copyOf(U[] original, int newLength, Class<? extends T[]> newType) {
        @SuppressWarnings("unchecked")
        //不管三元运算符的结果如何,都会创建一个新的数组
        //新数组的长度一定是和集合的size一样
        T[] copy = ((Object)newType == (Object)Object[].class)
            ? (T[]) new Object[newLength]
            : (T[]) Array.newInstance(newType.getComponentType(), newLength);
        //数组的拷贝
        System.arraycopy(original, 0, copy, 0,
                         Math.min(original.length, newLength));
        //返回新数组
        return copy;
    }
}

3.3 添加方法

方法名描述
public boolean add(E e)将指定的元素追加到此列表的末尾
public void add(int index, E element)在此列表中的指定位置插入指定的元素。
public boolean addAll(Collection<? extends E> c)按指定集合的Iterator返回的顺序将指定集合中的所有元素追加到此列表的末尾。
public boolean addAll(i nt index, Collection<? extends E> c)将指定集合中的所有元素插入到此列表中,从指定的位置开始。
  • public boolean add(E e) 添加单个元素
public class Test01 { 
    public static void main(String[] args) {
        ArrayList<String> list = new ArrayList<>(); 
        list.add("aaa"); 
    } 
}
  • 源代码
//将添加的数据传入给 e
public boolean add(E e) {
       //调用方法对内部容量进行校验
        ensureCapacityInternal(size + 1);
        elementData[size++] = e;
        return true;
    }

    private void ensureCapacityInternal(int minCapacity) {
        //判断集合存数据的数组是否等于空容量的数组
        if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
            //通过最小容量和默认容量 求出较大值 (用于第一次扩容)
            minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity);
        }
        //将if中计算出来的容量传递给下一个方法,继续校验
        ensureExplicitCapacity(minCapacity);
    }

    private void ensureExplicitCapacity(int minCapacity) {
            modCount++;

        if (minCapacity - elementData.length > 0)
            grow(minCapacity);
    }

    private void grow(int minCapacity) {
        //记录数组的实际长度,此时由于木有存储元素,长度为0
        int oldCapacity = elementData.length;
        // >> : 右移,右移几位就相当于除以2的几次幂
        // << : 左移,左移几位就相当于乘以2的几次幂
        //扩容的核心算法: 原容量的1.5倍
        int newCapacity = oldCapacity + (oldCapacity >> 1);
        //判断新容量 - 最小容量 是否小于 0, 如果是第一次调用add方法必然小于
        if (newCapacity - minCapacity < 0)
            //还是将最小容量赋值给新容量
            newCapacity = minCapacity;
        //判断新容量-最大数组大小 是否>0,如果条件满足就计算出一个超大容量
        if (newCapacity - MAX_ARRAY_SIZE > 0)
            newCapacity = hugeCapacity(minCapacity);
       // 调用数组工具类方法,创建一个新数组,将新数组的地址赋值给elementData
        elementData = Arrays.copyOf(elementData, newCapacity);
    }
 }
  • public void add(int index, E element)在指定索引处添加元素
public class AddDemo01 {
    public static void main(String[] args) {
        List<String> list=new ArrayList<>();
        list.add("张三");
        list.add("李四");
        list.add("王五");
        list.add(1,"赵六");
        System.out.println(list);
    }
}

源代码

 public void add(int index, E element) {
         //添加范围检查
        rangeCheckForAdd(index);
         //调用方法检验是否要扩容,且让增量++
        ensureCapacityInternal(size + 1);  // Increments modCount!!
        System.arraycopy(elementData, index, elementData, index + 1,
                         size - index);
        elementData[index] = element;
        size++;
    }

    private void ensureCapacityInternal(int minCapacity) {
        if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
            minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity);
        }

        ensureExplicitCapacity(minCapacity);
    }
     //确保明确的能力
     private void ensureExplicitCapacity(int minCapacity) {
        modCount++;

        //如果再调用 add(index,element) 方法之前已经扩容,那么源码跟踪到此结束 
        //不会进行扩容
        if (minCapacity - elementData.length > 0)
            grow(minCapacity);
     }

    private void rangeCheckForAdd(int index) {
        //超出指定范围就报错
        if (index > size || index < 0)
            throw new IndexOutOfBoundsException(outOfBoundsMsg(index));
    }
 }
   
  • public boolean addAll(Collection<? extends E> c)将集合的所有元素一次性添加到集合
public class AddDemo01 {
    public static void main(String[] args) {
        List<String> list=new ArrayList<>();
        List<String> list1=new ArrayList<>();
        list.add("张三");
        list.add("李四");
        list.add("王五");
        list.add(1,"赵六");
        list1.addAll(list);
        System.out.println(list1);
    }
}

控制台结果
[张三, 赵六, 李四, 王五]  

源码分析

public class ArrayList<E> {
    public boolean addAll(Collection<? extends E> c) {
        //把集合的元素转存到Object类型的数组中 
        Object[] a = c.toArray(); 
        //记录数组的长度
        int numNew = a.length; 
        //调用方法检验是否要扩容,且让增量++ 
        ensureCapacityInternal(size + numNew); 
        //调用方法将a数组的元素拷贝到elementData数组中
        System.arraycopy(a, 0, elementData, size, numNew); 
        //集合的长度+=a数组的长度 
        size += numNew; 
        //只要a数组的长度不等于0,即说明添加成功 
        return numNew != 0;
    } 
}
  • public boolean addAll(int index, Collection<? extends E> c)在指定的索引位置添加集合
public class AddDemo01 {
    public static void main(String[] args) {
        List<String> list=new ArrayList<>();
        List<String> list1=new ArrayList<>();
        list.add("张三");
        list.add("李四");
        list.add("王五");
        list.add(1,"赵六");
        list1.add("田七");
        list1.addAll(0,list);
        System.out.println(list1);
    }
}
控制台输出
[张三, 赵六, 李四, 王五, 田七]

源码分析

public class ArrayList<E> {
    public boolean addAll(int index, Collection<? extends E> c) {
        //校验索引
        rangeCheckForAdd(index);
        //将数据源转成数组
        Object[] a = c.toArray();
        //记录数据源的长度 4
        int numNew = a.length;
        //目的就是为了给集合存储数据的数组进行扩容
        ensureCapacityInternal(size + numNew);

        //numMoved:代表要移动元素的个数 --> 1个
        //numMoved: 数据目的(集合list1)的长度-调用addAll的第一个参数 (索引1)
        int numMoved = size - index;
        //判断需要移动的个数是否大于0
        if (numMoved > 0)
            //使用System中的方法arraycopy进行移动
            System.arraycopy(elementData, index, elementData, index + numNew,
                             numMoved);
        //才是真正将数据源(list)中的所有数据添加到数据目的(lsit1)
        System.arraycopy(a, 0, elementData, index, numNew);
        size += numNew;
        return numNew != 0;
    }

    private void rangeCheckForAdd(int index) {
        if (index > size || index < 0)
            throw new IndexOutOfBoundsException(outOfBoundsMsg(index));
    }
 }
}
public static void arraycopy(Object src,int srcPos,Object dest,int destPos,int length)
}
 src - 源数组。
 srcPos - 源数组中的起始位置。
 dest - 目标数组。
 destPos - 目的地数据中的起始位置。
 length - 要复制的数组元素的数量。

3.4 删除方法

  • public E remove(int index) 根据索引删除元素
public class Test01 {
    public static void main(String[] args) {
        ArrayList<String> list = new ArrayList<>(); 
        list.add("山东大李逵");
        list.add("天魁星宋江");
        list.add("天罡星卢俊义");
        list.add("西门大人");
        //根据索引删除元素 
        String value = list.remove(3);
        System.out.println("删除的元素为: "+value); 
        System.out.println("集合的元素: "+list); 
    } 
}

控制台输出
删除的元素为: 西门大人
集合的元素: [山东大李逵, 天魁星宋江, 天罡星卢俊义]    

源码分析

public class ArrayList<E> {
    public E remove(int index) {
        //范围校验
        rangeCheck(index); 
        //增量++ 
        modCount++; 
        //将index对应的元素赋值给 oldValue
        E oldValue = elementData(index); 
        //计算集合需要移动元素个数
        int numMoved = size - index - 1;
        //如果需要移动元素个数大于0,就使用arrayCopy方法进行拷贝
        //注意:数据源和数据目的就是elementData 
        if (numMoved > 0) 
            System.arraycopy(elementData, index+1, elementData, index, numMoved); 
        //将源集合最后一个元素置为null,尽早让垃圾回收机制对其进行回收 
        elementData[--size] = null; 
        //返回被删除的元素
        return oldValue;
    }
}
  • public boolean remove(Object o) 根据元素删除元素
public class Test01 {
    public static void main(String[] args) { 
        ArrayList<String> list = new ArrayList<>(); 
        list.add("山东大李逵"); 
        list.add("天魁星宋江"); 
        list.add("西门大人"); 
        list.add("天罡星卢俊义");
        //根据索引删除元素
        boolean flag = list.remove("西门大人"); 
        System.out.println("是否删除成功: "+flag);
        System.out.println("集合的元素: "+list);
    } 
}

控制台输出
是否删除成功: true
集合的元素: [山东大李逵, 天魁星宋江, 天罡星卢俊义]    

源码分析

public class ArrayList<E> {
    public boolean remove(Object o) {
        //判断要删除的元素是否为null
        if (o == null) { 
            //遍历集合
            for (int index = 0; index < size; index++) 
                //判断集合的元素是否为null 
                if (elementData[index] == null) { 
                    //如果相等,调用fastRemove方法快速删除
                    fastRemove(index);
                    return true;
                } 
        } else { 
            //遍历集合
            for (int index = 0; index < size; index++)
                //用o对象的equals方法和集合每一个元素进行比较
                if (o.equals(elementData[index])) { 
                    //如果相等,调用fastRemove方法快速删除
                    fastRemove(index);
                    return true;
                } 
        }
        //如果集合没有o该元素,那么就会返回false
        return false;
    }
    private void fastRemove(int index) {
        //增量++
        modCount++;
        //计算集合需要移动元素的个数
        int numMoved = size - index - 1; 
        //如果需要移动的个数大于0,调用arrayCopy方法进行拷贝
        if (numMoved > 0) System.arraycopy(elementData, index+1, elementData, index, numMoved); 
        //将集合最后一个元素置为null,尽早被释放 elementData[--size] = null;
    } 
}

3.5 修改方法

  • public E set(int index, E element)根据索引修改集合元素
public class Test01 {
    public static void main(String[] args) { 
        ArrayList<String> list = new ArrayList<>(); 
        list.add("山东大李逵");
        list.add("天魁星宋江"); 
        list.add("天罡星卢俊义");
        list.add("西门大人"); 
        //根据索引修改集合元素 
        String value = list.set(3, "花和尚鲁智深"); 
        System.out.println("set方法返回值: "+value); 
        System.out.println("集合的元素: "+list);
    } 
}

控制台输出
set方法返回值: 西门大人
集合的元素: [山东大李逵, 天魁星宋江, 天罡星卢俊义, 花和尚鲁智深]    

源码分析

public class ArrayList<E> {
    public E set(int index, E element) { 
        //范围校验 
        rangeCheck(index);
        //先取出index对应的元素,且赋值给oldValue
        E oldValue = elementData(index); 
        //将element直接覆盖index对应的元素
        elementData[index] = element; 
        //返回被覆盖的元素 
        return oldValue;
    }
    private void rangeCheck(int index) { 
        if (index >= size) throw new IndexOutOfBoundsException(outOfBoundsMsg(index)); 
    }
}

3.6 获取方法

  • public E get(int index)根据索引获取元素
public class Test01 {
    public static void main(String[] args) {
        ArrayList<String> list = new ArrayList<>(); 
        list.add("山东大李逵"); 
        list.add("天魁星宋江"); 
        list.add("天罡星卢俊义");
        list.add("西门大人"); 
        //根据索引获取集合元素 
        String value = list.get(1); 
        System.out.println("get方法返回值: "+value);
        System.out.println("集合的元素: "+list); 
    } 
}

控制台输出
get方法返回值: 天魁星宋江
集合的元素: [山东大李逵, 天魁星宋江, 天罡星卢俊义, 西门大人]    

源码分析

public class ArrayList<E> { 
    public E get(int index) {
        //范围校验 
        rangeCheck(index);
        //直接根据索引取出集合元素 
        return elementData(index); 
    }
    private void rangeCheck(int index) {
        if (index >= size) throw new IndexOutOfBoundsException(outOfBoundsMsg(index));
    }
}

3.7 转换方法

  • public String toString()把集合所有数据转换成字符串
public class Test01 {
    public static void main(String[] args) {
        ArrayList<String> list = new ArrayList<>(); 
        list.add("山东大李逵"); 
        list.add("天魁星宋江"); 
        list.add("天罡星卢俊义"); 
        list.add("西门大人"); 
        System.out.println("集合的元素: "+list); 
        //将集合的元素转换为字符串 
        String str = list.toString();
        System.out.println(str); 
    }
}

控制台输出
集合的元素: [山东大李逵, 天魁星宋江, 天罡星卢俊义, 西门大人]
[山东大李逵, 天魁星宋江, 天罡星卢俊义, 西门大人]    

源码分析

public class ArrayList<E> extends AbstractList<E>{ 
    public Iterator<E> iterator() {
        return new Itr(); 
    }//ArrayList集合内部类 
    private class Itr implements Iterator<E> { 
        int cursor; 
        int lastRet = -1; 
        //将实际修改集合次数赋值给预期修改次数 ,注意只会赋值一次
        //以后在迭代器获取元素的时候,每次都会判断集合实际修改次数是否和预期修改次数一致
        //如果不一致就会产生并发修改异常 
        int expectedModCount = modCount;
        //判断光标 和 集合的大小 是否不相等 
        public boolean hasNext() {
            return cursor != size; 
        }
        public E next() {
            checkForComodification();
            int i = cursor; 
            if (i >= size) throw new NoSuchElementException();
            Object[] elementData = ArrayList.this.elementData; 
            if (i >= elementData.length)
                throw new ConcurrentModificationException();
            cursor = i + 1; 
            return (E) elementData[lastRet = i];
        }
        final void checkForComodification() {
            if (modCount != expectedModCount) 
                throw new ConcurrentModificationException();
        } 
    } 
}
public abstract class AbstractCollection<E> {
    public String toString() {
        //获取迭代器
        Iterator<E> it = iterator();
        //判断迭代器是否有元素
        if (! it.hasNext())
            return "[]";
        //创建StringBuilder
        StringBuilder sb = new StringBuilder();
        //先追加了'['
        sb.append('[');
        //无限循环
        for (;;) {
            //调用迭代器的next方法取出元素,且将光标向下移动
            E e = it.next();
            //三元判断
            sb.append(e == this ? "(this Collection)" : e);
            if (! it.hasNext())
                //没有元素,在缓冲区的最后追加']',且把整个缓冲区的数据转成字符串
                //然后再介绍该方法
                return sb.append(']').toString();

            //有元素,就直接追加
            sb.append(',').append(' ');
        }
    }

3.8 迭代器

  • public Iterator<E> iterator()普通迭代器

源码同上(在讲toString方法的时候已经讲过基本操作,通过以下两个案例进行一步分析源码)

案例一: 已知集合:List list = new ArrayList();里面有三个元素:“hello”、“Java”、“PHP”,使用迭代器遍

历获取集合的每一个元素

public class Test01 {
    public static void main(String[] args) {
        //创建集合对象 
        List<String> list = new ArrayList<String>(); 
        //添加元素 
        list.add("hello");
        list.add("Java");
        list.add("PHP");
        //获取迭代器
        Iterator<String> it = list.iterator();
        //遍历集合 
        while (it.hasNext()) { 
            String s = it.next();
            System.out.println(s); 
        }
    }
}

案例二: 已知集合:List list = new ArrayList();里面有三个元素:“hello”、“Java”、“PHP”,使用迭代器遍

历集合看有没有"PHP"这个元素,如果有,就使用集合对象删除该元素

public class Test01 { 
    public static void main(String[] args) {
        //创建集合对象 
        List<String> list = new ArrayList<String>(); 
        //添加元素
        list.add("hello"); 
        list.add("Java");
        list.add("PHP"); 
        //获取迭代器 
        Iterator<String> it = list.iterator();
        //遍历集合 
        while (it.hasNext()) {
            String s = it.next();
            if(s.equals("PHP")) {
                list.remove("PHP");
            }
        } 
    } 
}
控制台输出:并发修改异常
Exception in thread "main" java.util.ConcurrentModificationException at
java.util.ArrayList$Itr.checkForComodification(ArrayList.java:901) at
java.util.ArrayList$Itr.next(ArrayList.java:851) at cn.itheima.method.Test01.main(Test01.java:24)    

源码分析:(应该从获取迭代器的时候就进入到源代码中)

//获取迭代器的方法
    public Iterator<E> iterator() {
        //创建了一个对象
        return new Itr();
    }

    //ArrayList集合的内部类 --> 迭代器的源码
    private class Itr implements Iterator<E> {
        int cursor;       // 光标,默认值就是0
        int lastRet = -1; // 记录-1
        // 将集合实际修改次数赋值给预期修改次数
        // 获取迭代器的时候,那么expectedModCount的值也就是 3
        int expectedModCount = modCount;

        //判断集合是否有元素
        public boolean hasNext() {
            //光标是否不等于集合的size 3
            return cursor != size;
        }

        public E next() {
            checkForComodification();
            //光标赋值给i = 0
            int i = cursor;
            //判断,如果大于集合的size就说明没有元素了
            if (i >= size)
                throw new NoSuchElementException();
            //把集合存储数据数组的地址赋值给该方法的局部变量
            Object[] elementData = ArrayList.this.elementData;
            //进行判断,如果条件满足就会产生并发修改异常
            if (i >= elementData.length)
                throw new ConcurrentModificationException();
            //光标自增
            cursor = i + 1;
            //从数组中取出元素且返回
            return (E) elementData[lastRet = i];
        }

        //校验预期修改集合次数是否和实际修改集合次数一样
        final void checkForComodification() {
            if (modCount != expectedModCount)
                //如果不一样,就会产生并发修改异常
                throw new ConcurrentModificationException();
        }
    }

    //集合删除元素的方法
    public boolean remove(Object o) {
        //判断要删除的元素是否为null
        if (o == null) {
            for (int index = 0; index < size; index++)
                if (elementData[index] == null) {
                    fastRemove(index);
                    return true;
                }
        } else {
            //遍历集合
            for (int index = 0; index < size; index++)
                //拿着要删除的元素和集合的每一个元素进行比较
                if (o.equals(elementData[index])) {
                    //如果相等就调用方法进行删除
                    fastRemove(index);
                    return true;
                }
        }
        return false;
    }

    //真正删除元素的方法
    private void fastRemove(int index) {
        //在删除的方法中集合实际修改次数会自增
        //集合实际修改次数为:4 但是预期修改次数为:3
        modCount++;
        //计算集合要移动元素的个数
        int numMoved = size - index - 1;
        if (numMoved > 0)
            //移动的核心代码
            System.arraycopy(elementData, index+1, elementData, index,
                             numMoved);
        //就是让删除的元素置为null,就是为了尽快被垃圾回收机制回收
        elementData[--size] = null; // clear to let GC do its work
    }

 }

 结论:,集合每次调用remove方法的时候,实际修改次数变量的值都会自增一次
    二,在获取迭代器的时候,集合只会执行一次将实际修改集合的次数赋值给预期修改集合的次数
    三,集合在删除元素的时候也会针对实际修改次数的变量进行自增的操作

案例三:已知集合:List list = new ArrayList();里面有三个元素:“hello”、“PHP”、“JavaSE”,使用迭代器
遍历集合看有没有"PHP"这个元素,如果有,就使用集合对象删除该元素

public class Test01 {
    public static void main(String[] args) {
        //创建集合对象 
        List<String> list = new ArrayList<String>();
        //添加元素
        list.add("hello");
        list.add("PHP"); 
        list.add("Java");
        //获取迭代器
        Iterator<String> it = list.iterator(); 
        //遍历集合
        while (it.hasNext()) {
            String s = it.next();
            if(s.equals("PHP")) {
                list.remove("PHP");
            } 
        }
        System.out.println(list);
    } 
}
控制台输出
[hello, Java]    
迭代删除,在倒数第二个的不会导致错误    

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-TLNEZ637-1623374937348)(ArrayList底层源码分析.assets/image-20201227102656677.png)]

  • default void remove() 迭代器中的remove方法,删除集合中的元素
public class Test01 {
    public static void main(String[] args) { 
        //创建集合对象 
        List<String> list = new ArrayList<String>();
        //添加元素
        list.add("hello");
        list.add("PHP");
        list.add("Java"); 
        //获取迭代器 
        Iterator<String> it = list.iterator(); 
        //遍历集合 
        while (it.hasNext()) { 
            String s = it.next();
            if(s.equals("hello")) {
                it.remove(); 
            }
        }
        System.out.println(list);
    }
}
控制台输出
[PHP, Java]    

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-8xD6n0X7-1623374937352)(ArrayList底层源码分析.assets/image-20201227103529517.png)]

源码分析**(应该从获取迭代器的时候就进入到源代码中)**

public class ArrayList<E> { 
    public Iterator<E> iterator() { 
        return new Itr();
    }
    //ArrayList内部类
    //一定要注意观察 Itr 类中的几个成员变量
    private class Itr implements Iterator<E> { 
        int cursor; 
        // 下一个要返回元素的索引
        int lastRet = -1;
        // 最后一个返回元素的索引
        //将实际修改集合次数 赋值 给预期修改次数
        //在迭代的过程中,只要实际修改次数和预期修改次数不一致就会产生并发修改异常
        //由于expectedModCount是Itr的成员变量,那么只会被赋值一次!!! 
        //同时由于集合调用了三次add方法,那么实际修改集合次数就是 3,因此expectedModCount的值也是 3   
        int expectedModCount = modCount;
        public boolean hasNext() { 
            return cursor != size;
        }
        //获取元素的方法
        public E next() { 
            checkForComodification();
            //把下一个元素的索引赋值给i 
            int i = cursor; 
            //判断是否有元素 
            if (i >= size) throw new NoSuchElementException(); 
            //将集合底层存储数据的数组赋值给迭代器的局部变量 elementData
            Object[] elementData = ArrayList.this.elementData; 
            //再次判断,如果下一个元素的索引大于集合底层存储元素的长度 并发修改异常 
            //注意,尽管会产生并发修改异常,但是这里显示不是我们要的结果
            if (i >= elementData.length) throw new ConcurrentModificationException();
            //每次成功获取到元素,下一个元素的索引都是当前索引+1 
            cursor = i + 1; 
            //返回元素,且将i的值 赋值给 lastRet /*0*/ 
            return (E) elementData[lastRet = i];
        }
        //迭代器删除元素方法 
        public void remove() {
            //判断最后返回元素的索引是否小于0,满足条件就产生 非法状态异常
            if (lastRet < 0) throw new IllegalStateException();
            //校验是否会产生并发修改异常,第一次调用不会,因为与其修改次数和实际修改次数一致
            checkForComodification();
            try {
                //真正删除集合元素的方法,调用方法为ArrayList的方法remove,且将0作为参数进行传递 
                ArrayList.this.remove(lastRet); 
                //将lastRet赋值给cursor
                cursor = lastRet; 
                //再次等于-1 
                lastRet = -1;
                //再次将集合实际修改次数赋值给预期修改次数,那么这个时候不管集合自身是否删除成功
                //那么实际修改次数和预期修改次数又一致了,所以并不会产生并发修改异常
                expectedModCount = modCount; 
            } catch (IndexOutOfBoundsException ex) { 
                throw new ConcurrentModificationException();
            }
        }
        final void checkForComodification() { 
            //如果预期修改次数 和 实际修改次数不相等 就产生并发修改异常
            if (modCount != expectedModCount) throw new
                ConcurrentModificationException();
        } 
    }
    //集合的remove方法 
    public boolean remove(Object o) {
        if (o == null) { 
            for (int index = 0; index < size; index++) 
                if (elementData[index] == null) {
                    fastRemove(index); return true;
                } 
        } else {
            for (int index = 0; index < size; index++) 
                if (o.equals(elementData[index])) { 
                    fastRemove(index); 
                    return true;
                    }
        }return false; 
    }
    //快速删除方法 
    private void fastRemove(int index) { 
        //最最最关键的一个操作,集合实际修改次数++,那么这个时候由原来的3变成4
        //but迭代器的预期修改次数还是3!!! 
        modCount++;
        int numMoved = size - index - 1;
        if (numMoved > 0) 
            System.arraycopy(elementData, index+1, elementData, index, numMoved);
        //还有一个很关键的操作,集合的长度也发生了改变 
        elementData[--size] = null;
    } 
}
结论:
 1、迭代器remove方法底层调用的还是集合自身的remove方法删除元素
 2、之所以不会产生并发修改异常,其原因是因为在迭代器的remove方法中会再次将集合时机修改次数赋值
给预期修改次数

3.9 清空方法

  • public void clear()清空集合所有数据
public class Test01 {
    public static void main(String[] args) { 
        //创建集合对象
        List<String> list = new ArrayList<String>();
        //添加元素 
        list.add("hello");
        list.add("PHP");
        list.add("Java");
        System.out.println("清空前的集合: "+list); 
        //清空集合所有元素 
        list.clear(); 
        System.out.println("清空后的集合: "+list); 
    } 
}

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-tZsmulPT-1623374937354)(ArrayList底层源码分析.assets/image-20201227104118690.png)]

源码分析

public class ArrayList<E> {
    public void clear() {
        //实际修改集合次数++ 
        modCount++; 
        //遍历集合,将集合每一个索引对应位置上的元素都置为null,尽早让其释放 
        for (int i = 0; i < size; i++) 
            elementData[i] = null; 
        //集合长度更改为0 
        size = 0; 
    } 
}

3.10 包含方法

  • public boolean contains(Object o)判断集合是否包含指定元素
public class Test01 {
    public static void main(String[] args) { 
        //创建集合对象
        List<String> list = new ArrayList<String>(); 
        //添加元素
        list.add("hello");
        list.add("PHP");
        list.add("Java");
        System.out.println("判断之前集合的元素: "+list);
        //需求:如果集合中没有JavaSE该元素,请添加一个JavaSE元素
        //解决方式一:循环遍历集合,判断集合是否包含JavaSE,如果没有包含就调用集合的add方法进行添加操作 
        //解决方式二:使用集合contains方法判断,根据判断的结果决定是否要添加元素 
        if(!list.contains("JavaSE")){ 
            list.add("JavaSE");
        }
        System.out.println("判断之后集合的元素: "+list);
    } 
}

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-KaAuqXVj-1623374937356)(ArrayList底层源码分析.assets/image-20201227104428298.png)]

源码分析

public class ArrayList<E> { 
    //源码contains方法
    public boolean contains(Object o) { 
        //调用indexOf方法进行查找
        return indexOf(o) >= 0; 
    }
    public int indexOf(Object o) { 
        //如果元素是null,也进行遍历操作
        //因为集合中有可能够会存储null
        if (o == null) {
            for (int i = 0; i < size; i++) 
                if (elementData[i]==null) return i;
        } else 
        { 
            for (int i = 0; i < size; i++) 
                if (o.equals(elementData[i]))
                    return i; 
        }
        //如果没有走if,也没有走else,那么就说明o该元素在集合中不存在 
        return -1;
    } 
}
结论:底层也是通过循环遍历集合,取出一个个的元素和要找的元素进行比较

3.11 判断集合是否为空

public class Test01 {
    public static void main(String[] args) {
        //创建集合对象 
        List<String> list = new ArrayList<String>(); 
        //添加元素
        list.add("hello");
        list.add("PHP");
        list.add("Java");
        boolean b = list.isEmpty();
        System.out.println(b);
        System.out.println(list);
    } 
}

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-f96g5NiP-1623374937362)(ArrayList底层源码分析.assets/image-20201227104652806.png)]

源码分析

public class ArrayList<E> { public boolean isEmpty() { return size == 0; } }

4. 面试题

4.1 ArrayList是如何扩容的?

private void grow(int minCapacity) {
        int oldCapacity = elementData.length;
        int newCapacity = oldCapacity + (oldCapacity >> 1);
        if (newCapacity - minCapacity < 0)
            newCapacity = minCapacity;
        if (newCapacity - MAX_ARRAY_SIZE > 0)
            newCapacity = hugeCapacity(minCapacity);
        elementData = Arrays.copyOf(elementData, newCapacity);
    }
第一次扩容10,以后每次都是原容量的1.5,在执行add方法时候扩容,而不是在初始化时

4.2 ArrayList频繁扩容导致添加性能急剧下降,如何处理?

  • 案例
public class Test01 {
    public static void main(String[] args) { 
        //创建集合对象
        List<String> list = new ArrayList<String>(); 
        //添加元素
        list.add("hello"); 
        list.add("PHP");
        list.add("Java");
        long startTime = System.currentTimeMillis();
        //需求:还需要添加100W条数据 
        for (int i = 0; i < 1000000; i++) { 
            list.add(i+"");
        }
        long endTime = System.currentTimeMillis(); 
        System.out.println("未指定容量: "+ (endTime - startTime)+"ms");
    } 
}
控制台输出
未指定容量: 123ms
  • 解决方法
public class Test01 {
    public static void main(String[] args) {
        //创建集合对象,并指定初始容量10000
        List<String> list = new ArrayList<>(10000);
        //添加元素
        list.add("hello");
        list.add("PHP");
        list.add("Java");
        long startTime = System.currentTimeMillis();
        //需求:还需要添加100W条数据
        for (int i = 0; i < 1000000; i++) {
            list.add(i + "");
        }
        long endTime = System.currentTimeMillis();
        System.out.println("指定容量: " + (endTime - startTime)+"ms");
    }
}
控制台输出
指定容量: 117ms

4.3 ArrayList插入或删除元素一定比LinkedList慢么?

  • 根据索引删除

案例:ArrayList和LinkedList对比

public class Test02 {    public static void main(String[] args) {        //创建ArrayList集合对象        ArrayList<String> arrayList = new ArrayList<String>();        //添加500W个元素         for (int i = 0; i < 5000000; i++) {             arrayList.add(i+"ljt");        }        //获取开始时间         long startTime = System.currentTimeMillis();         //根据索引删除ArrayList集合元素        //删除索引5000对应的元素        String value = arrayList.remove(50000);         System.out.println(value);         //获取结束时间        long endTime = System.currentTimeMillis();         System.out.println("ArrayList集合删除元素的时间: "+(endTime-startTime)+"ms");         //创建LinkedList集合对象         LinkedList<String> linkedList = new LinkedList<String>();         //添加500W个元素        for (int i = 0; i < 5000000; i++) {            linkedList.add(i+"ljt");         }        //获取开始时间        startTime = System.currentTimeMillis();         //根据索引删除LinkedList集合元素         //删除索引5000对应的元素         value = arrayList.remove(50000);        System.out.println(value);        endTime = System.currentTimeMillis();        System.out.println("LinkedList集合删除元素的时间: "+(endTime-startTime)+"ms");    }}控制台输出500000ljtArrayList集合删除元素的时间: 3ms500001ljtLinkedList集合删除元素的时间: 3ms    

源码分析

  • ArrayList根据索引删除元素源码
public class ArrayList<E> { 
    public E remove(int index) { 
        //范围校验
        rangeCheck(index);
        //增量++ 
        modCount++;
        //将index对应的元素赋值给 
        oldValue E oldValue = elementData(index); 
        //计算集合需要移动元素个数 
        int numMoved = size - index - 1;
        //如果需要移动元素个数大于0,就使用arrayCopy方法进行拷贝 
        //注意:数据源和数据目的就是elementData
        if (numMoved > 0) 
            System.arraycopy(elementData, index+1, elementData, index, numMoved); 
        //将源集合最后一个元素置为null,尽早让垃圾回收机制对其进行回收
        elementData[--size] = null;
        //返回被删除的元素 return oldValue; 
    } 
}
  • LinkedList根据索引删除元素源码
public class LinkedList<E> { 
    public E remove(int index) { 
        //调用方法校验元素的索引 
        checkElementIndex(index); 
        //先调用node(index)方法,找到需要删除的索引 
        //再调用unlink方法解开链条
        return unlink(node(index));
    }
    //校验索引是否在合法范围之内,不再就报错 
    private void checkElementIndex(int index) {
        if (!isElementIndex(index)) throw new IndexOutOfBoundsException(outOfBoundsMsg(index)); 
    }
    private boolean isElementIndex(int index) {
        return index >= 0 && index < size;
    }
    //获取要删除的元素 
    Node<E> node(int index) { 
        //不管索引是多少,在源码底层都会对整个链表上的元素进行折半的动作 
        //如果要删除元素的索引小于集合长度的一半,那么就从头节点一个个的往后找 
        //如果要删除元素的索引大于集合长度的一半,那么就从尾节点一个个的往后找
        //(注:这个查找的效率相对于ArrayList集合来说较低) 
        if (index < (size >> 1)) {
            Node<E> x = first;
            //如果循环条件不满足,那么first就是要删除的元素
            //否则,要删除的元素就是first的下一个
            for (int i = 0; i < index; i++) 
                x = x.next;
            return x; }
        else { Node<E> x = last;
              //如果循环条件不满足,那么last就是要删除的元素
              //否则,要删除的元素就是last的前一个
              for (int i = size - 1; i > index; i--) 
                  x = x.prev; return x; 
             }
    }
    //解开链表,让前后节点相互记录地址
    E unlink(Node<E> x) {
        //获取要删除的元素
        final E element = x.item;
        //获取被删除节点下一个节点的地址 
        final Node<E> next = x.next;
        //获取被删除节点上一个节点的地址 
        final Node<E> prev = x.prev; 
        //如果被删除节点的上一个节点为null,就让被删除节点的下一个节点成为首节点 
        if (prev == null) {
            first = next;
        } else { 
            //否则,被删除元素上一个节点的 下一个节点 变成 被删除元素的下一个节点 
            prev.next = next; 
            //被删除元素的上一个节点置为null
            x.prev = null;
        }
        //如果被删除元素的下一个节点为null,最后一个节点就等于被删除元素的上一个节点
        if (next == null) { 
            last = prev; 
        } else {
            //否则,被删除节点的下一个节点 等于被删除节点的前一个节点 
            next.prev = prev; 
            //被删除元素的下一个节点置为null
            x.next = null; 
        }//被删除元素的内容置为null 
        x.item = null; 
        //集合长度--
        size--; 
        //实际修改次数++ 
        modCount++; 
        //返回被删除的元素 
        return element;
    } 
}
结论
1.数组删除元素确实要比链表慢,慢在需要创建新数组,还有比较麻烦的数据拷贝,但是在ArrayList
底层不是每次删除元素都需要扩容,因此在这个方面相对于链表来说数组的性能更好
2. LinkedList删除元素之所以效率并不高,其原理在于底层先需要对整个集合进行折半的动作,然后
又需要对集合进行遍历一次,这些操作导致效率变低    
  • ArrayList和LinkedList插入元素
public class TestLinkAndArray {
    public static void main(String[] args) {
        //创建ArrayList集合对象
        ArrayList<String> arrayList = new ArrayList<>();
        //添加500W个元素
        //获取开始时间
        long startTime = System.currentTimeMillis();
        for (int i = 0; i < 5000000; i++) {
            arrayList.add(i + "ljt");
        }
        long endTime = System.currentTimeMillis();

        System.out.println("ArrayList集合插入元素的时间: " + (endTime - startTime) + "ms");
        //创建LinkedList集合对象
        LinkedList<String> linkedList = new LinkedList<>();
        //添加500W个元素
        //获取开始时间
        startTime = System.currentTimeMillis();
        for (int i = 0; i < 5000000; i++) {
            linkedList.add(i + "ljt");
        }
        endTime = System.currentTimeMillis();
        System.out.println("LinkedList集合插入元素的时间: " + (endTime - startTime) + "ms");
    }
}
控制台输出
ArrayList集合插入元素的时间: 2724ms
LinkedList集合插入元素的时间: 2081ms

结论:插入元素LinkedList明显比ArrayList快,而且数据越多,越明显    

4.4 ArrayList是线程安全的么?

  • ArrayList不是线程安全的
//线程任务类
public class CollectionTask implements Runnable {
    //通过构造方法共享一个集合
    private List<String> list;
    public CollectionTask(List<String> list) {
        this.list = list;
    }
    @Override 
    public void run() {
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) { 
            e.printStackTrace();
        }
        //把当前线程名字加入到集合 
        list.add(Thread.currentThread().getName()); 
    } 
}
//测试类
 class CollectionTest01 {
    public static void main(String[] args) throws InterruptedException { 
        //创建集合
        List<String> list = new ArrayList<String>(); 
        //创建线程任务
        CollectionTask ct = new CollectionTask(list);
        //开启50条线程 
        for (int i = 0; i < 50; i++) { 
            new Thread(ct).start();
        }
        //确保子线程执行完毕 
        Thread.sleep(1000); 
        //遍历集合
        for (int i = 0; i < list.size(); i++) {
            System.out.println(list.get(i));
        }
        System.out.println("集合长度: "+list.size()); 
    } 
}

控制台输出
Thread-26
Thread-24
Thread-23
null
集合长度: 21    
  • 解决方案

    1.在run方法中添加同步代码块synchronized(this)
    2.list = Collections.synchronizedList(list);    
    
  • 实际开发场景

案例:使用JdbcTemplate查询数据库返回一个List集合是否需要保证线程安全?

不需要,因为list是局部变量,即是线程不共享,线程安全
public class Test01 { 
    //创建JdbcTemplate对象
    JdbcTemplate jt = new JdbcTemplate(JDBCUtils.getDataSource());
    //3.利用JDBC查询出基础班在读的男学员的所有信息按成绩的降序输出到控制台上(利用JDBC) 
    @Test public void fun1() throws Exception {
        //拼写SQL 
        String sql = "select * from stutb where sex = ? and type like ? order by score desc";
        //调用方法查询 将结果集的每一行都封装成一个Stutb对象,并把每一个对象都添加到集合 
        //查询的结果是否需要保证线程安全??? 
        List<Stutb> list = jt.query(sql, new BeanPropertyRowMapper<Stutb>(Stutb.class), "男", "%基础班%");
        //在遍历集合取出结果集之前面临一个问题,使用普通for遍历好 还是使用迭代器(增强for)? //特别是数据量特别大的时候一定要考虑! //对返回的集合进行判断,如果返回的集合实现了 RandomAccess 就使用 普通for
        //否则使用迭代器(增强for)
        if(list instanceof RandomAccess){ 
            for (int i = 0; i < list.size(); i++) {
                System.out.println(list.get(i));
            } 
        }else { 
            for (Stutb stutb : list) { 
                System.out.println(stutb);
            }
        }
    }
}

4.5 如何复制某个ArrayList到另一个ArrayList中去?

  • 使用clone()方法
  • 使用ArrayList构造方法
  • 使用addAll方法

4.6 已知成员变量集合存储N多用户名称,在多线程的环境下,使用迭代器在读取集合数据的同时如何保证还可以正常的写入数据到集合?

普通集合 ArrayList

//线程任务类 
class CollectionThread implements Runnable{ 
    private static ArrayList<String> list = new ArrayList<String>();
    static{
           list.add("Jack"); 
           list.add("Lucy");
           list.add("Jimmy");
          }
    @Override 
    public void run() { 
        for (String value : list) {
            System.out.println(value); 
            //在读取数据的同时又向集合写入数据
            list.add("coco");
        } 
    } 
}
//测试类 
public class ReadAndWriteTest {
    public static void main(String[] args) { 
        //创建线程任务
        CollectionThread ct = new CollectionThread(); 
        //开启10条线程
        for (int i = 0; i < 10; i++) { 
            new Thread(ct).start();
        } 
    } 
}
  • 并发修改异常

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-S0m0Y7i3-1623374937363)(ArrayList底层源码分析.assets/image-20201227114625416.png)]

  • 解决方法
使用 
private static CopyOnWriteArrayList<String> list = new CopyOnWriteArrayList<>();

4.7 ArrayList LinkList区别?

  • ArrayList

    • 基于动态数组的数据结构

    • 对于随机访问的get和set,ArrayList要优于LinkedList

    • 对于随机操作的add和remove,ArrayList不一定比LinkedList慢 (ArrayList底层由于是动态数组,因此

      并不是每次add和remove的时候都需要创建新数组)

  • LinkedList

    • 基于链表的数据结构
    • 对于顺序操作,LinkedList不一定比ArrayList慢
    • 对于随机操作,LinkedList效率明显较低

4.8 自定义ArrayList

public class MyArrayList<E> {
    //定义数组,用于存储集合的元素
    private Object[] elementData;
    //定义变量,用于记录数组的个数
    private int size;
    //定义空数组,用于在创建集合对象的时候给elementData初始化
    private Object[] emptyArray = {};
    //定义常量,用于记录集合的容量
    private final int DEFAULT_CAPACITY = 10;

    //构造方法
    public MyArrayList() {
        //给elementData初始化
        elementData = emptyArray;
    }

    //定义add方法
    public boolean add(E element) {
        grow();
        elementData[size++] = element;
        return true;
    }

    //扩容
    private void grow() {
        if (elementData == emptyArray) {
            elementData = new Object[DEFAULT_CAPACITY];
        }
        if (elementData.length == size) {
            int oldCapacity = elementData.length;
            int newCapacity = oldCapacity + (oldCapacity >> 1);
            Object[] obj = new Object[newCapacity];
            System.arraycopy(elementData, 0, obj, 0, elementData.length);
            elementData = obj;
        }
    }

    //toString方法
    public String toString() {
        if (size == 0) return "[]";
        StringBuilder sb = new StringBuilder();
        sb.append("[");
        for (int i = 0; i < size; i++) {
            if (i == size - 1) {
                sb.append(elementData[i]).append("]");
            } else {
                sb.append(elementData[i]).append(", ");
            }
        }
        return sb.toString();
    }

    //修改方法
    public E set(int index, E element) {
        checkIndex(index);
        E oldValue = (E) elementData[index];
        elementData[index] = element;
        return oldValue;
    }


    private void checkIndex(int index) {
        if (index < 0 || index >= size)
            throw new IndexOutOfBoundsException("数组越界异常");
    }

    //删除方法
    public E remove(int index) {
        checkIndex(index);
        E oldValue = (E) elementData[index];
        int numMoved = size - index - 1;
        if (numMoved > 0) {
            System.arraycopy(elementData, index + 1, elementData, index, numMoved);
        }
        elementData[--size] = null;
        return oldValue;
    }

    //获取元素方法
    public E get(int index) {
        checkIndex(index);
        return (E) elementData[index];
    }

    //获取长度方法
    public int getSize() {
        return size;
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值