全网最全ArrayList底层原理实现

1. ArrayList集合底层数据结构

1. ArrayList集合介绍

ArrayList是实现了List接口的动态数组,所谓动态数组就是他的大小是可变的。实现了所有可选列表操作,并允许包括Null在内的所有元素。除了实现 List 接口外,此类还提供一些方法来操作内部用来存储列表的数组的大小。

每个ArrayList实例都有一个容量,该容量是指用来存储列表元素的数组的大小。默认初始容量是10。默认初始容量为10。随着ArrayList中元素的增加,它的容量也会不断的自动增长。在每次添加元素时,ArrayList都会检查是否需要进行扩容操作,扩容操作带来数据向新数组的重新拷贝,所以如果我们知道具体业务数据量,在构造ArrayList时,可以给ArrayList 指定一个初始容量,这样就会减少扩容时的拷贝问题。当然在添加大量元素前,应用程序也可以使用ensureCapacity操作来增加ArrayList实例的容量,这可以减少递增式再分配的数量。

ArrayList 是大家最为常用的集合类,我们先来看下常用的方法:

ArrayList是最常用的集合,底层是用数组实现的,继承AbstractList类,实现了List,RandomAccess,Cloneable和Serializable接口,非线程安全。

List<String> dataList = new ArrayList<>();//创建 ArrayList
dataList.add("test");//添加数据
dataList.add(1,"test1");//指定位置,添加数据
dataList.get(0);//获取指定位置的数据
dataList.remove(0);//移除指定位置的数据
dataList.clear();//清空数据
  • List 接口的可调整大小的数组实现。
  • 数组:一旦初始化长度就不可以发生改变

2. 数组结构介绍

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

2. ArrayList继承关系

2.1 Serializable标记性接口

  1. 介绍 类的序列化由实现java.io.Serializable接口的类启用。 不实现此接口的类将不会使任何状态序列化或反序列化。 可序列化类的所有子类型都是可序列化的。 序列化接口没有方法或字段,仅用于标识可串行化的语义。
  • 序列化:将对象的数据写入到文件(写对象)
  • 反序列化:将文件中对象的数据读取出来(读对象)
  1. Serializable源码介绍
public interface Serializable {
}

案例: 通过序列化流序列化和反序列化集合

package com.maweiqi.pojo;public class Student  {//姓名private String name;//年龄private Integer age;public Student() {}public Student(String name, Integer age) {this.name = name;this.age = age;}@Overridepublic String toString() {//优化toString方法StringBuilder sb = new StringBuilder();sb.append("[").append("name = ").append(this.name).append(", ").append("age =").append(this.age).append("]");return sb.toString();}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 class Test01 {public static void main(String[] args) throws Exception{Student s = new Student();
//创建对象操作流 --> 序列化ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("test\\obj.txt"));
//创建集合,且添加学生对象ArrayList<Student> list = new ArrayList<Student>();list.add(new Student("张三",51));list.add(new Student("李四",26));list.add(new Student("王五",32));list.add(new Student("赵六",27));
//将集合写入到文件oos.writeObject(list);
//创建对象输入流 --> 反序列化ObjectInputStream ois = new ObjectInputStream(new FileInputStream("test\\obj.txt"));
//读取数据Object o = ois.readObject();
//向下转型ArrayList<Student> al = (ArrayList<Student>) o;
//遍历集合for (int i = 0; i < al.size(); i++) {
//根据索引取出集合的每一个元素Student stu = al.get(i);System.out.println(stu);}}
}

运行程序

Exception in thread "main" java.io.NotSerializableException: com.maweiqi.pojo.Studentat java.io.ObjectOutputStream.writeObject0(ObjectOutputStream.java:1184)at java.io.ObjectOutputStream.writeObject(ObjectOutputStream.java:348)at java.util.ArrayList.writeObject(ArrayList.java:768)at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)at java.lang.reflect.Method.invoke(Method.java:498)at java.io.ObjectStreamClass.invokeWriteObject(ObjectStreamClass.java:1155)at java.io.ObjectOutputStream.writeSerialData(ObjectOutputStream.java:1496)at java.io.ObjectOutputStream.writeOrdinaryObject(ObjectOutputStream.java:1432)at java.io.ObjectOutputStream.writeObject0(ObjectOutputStream.java:1178)at java.io.ObjectOutputStream.writeObject(ObjectOutputStream.java:348)at com.maweiqi.Test01.main(Test01.java:24)

Student 需要实现序列化接口

public class Student  implements Serializable {
}

运行程序 查看项目的根目录,生成序列化和反序列化文件

Student{name='张三', age=51}
Student{name='李四', age=26}
Student{name='王五', age=32}
Student{name='赵六', age=27}

2.2 Cloneable 标记性接口

  1. 介绍 一个类实现 Cloneable 接口来指示 Object.clone() 方法,该方法对于该类的实例进行字段的复制是合法的。在不实现 Cloneable 接口的实例上调用对象的克隆方法会导致异常 CloneNotSupportedException 被抛出。简言之:克隆就是依据已经有的数据,创造一份新的完全一样的数据拷贝
  2. Cloneable源码介绍
public interface Cloneable {
}
  1. 克隆的前提条件
  • 被克隆对象所在的类必须实现 Cloneable 接口
  • 必须重写 clone 方法
  1. clone的基本使用
/*** 克隆的基本使用:* 将ArrayList集合的数据clone到另外一个集合*/
public class ArrayList_Clone {public static void main(String[] args) {ArrayList<String> list = new ArrayList<String>();list.add("人生就是旅途");list.add("也许终点和起点会重合");list.add("但是一开始就站在起点等待终点");list.add("那么其中就没有美丽的沿途风景和令人难忘的过往");
//调用方法进行克隆Object o = list.clone();System.out.println(o == list);System.out.println(o);System.out.println(list);}
}

运行程序

false
[人生就是旅途, 也许终点和起点会重合, 但是一开始就站在起点等待终点, 那么其中就没有美丽的沿途风景和令人难忘的过往]
[人生就是旅途, 也许终点和起点会重合, 但是一开始就站在起点等待终点, 那么其中就没有美丽的沿途风景和令人难忘的过往]
  1. clone源码分析
public class ArrayList<E> {public Object clone() {try {ArrayList<?> v = (ArrayList<?>) super.clone();v.elementData = Arrays.copyOf(elementData, size);v.modCount = 0;return v;} catch (CloneNotSupportedException e) {throw new InternalError(e);}}
}

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

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

方式一:创建两个对象模拟
6. 准备学生类

public class Student   {//姓名private String name;//年龄private Integer age;public Student() {}public Student(String name, Integer age) {this.name = name;this.age = age;}@Overridepublic String toString() {return "Student{" +"name='" + name + '\'' +", 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;}
}
  1. 准备测试类
/**** 案例:已知A对象的姓名为豹子头林冲,年龄30 。由于项目特殊要求需要将该对象的数据* 复制另外一个对象B中,并且此后A和B两个对象的数据不会相互影响*/
@SuppressWarnings("all")
public class CloneTest01 {public static void main(String[] args) {//传统方式://创建学生对象Student stu1 = new Student("豹子头林冲",30);//再次创建一个新的学生对象Student stu2 = new Student();//把stu1对象name的值取出来赋值给stu2对象的namestu2.setName(stu1.getName());//把stu1对象age的值取出来赋值给stu2对象的agestu2.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);}
}
  1. 控制台效果
false
Student{name='豹子头林冲', age=30}
Student{name='豹子头林冲', age=30}
----此时不管修改哪个对象的内容,stu1和stu2都不会受到影响----
Student{name='扈三娘', age=30}
Student{name='豹子头林冲', age=30}

方式二:使用克隆
浅拷贝克隆
修改Student 增加克隆方法

public class Student  implements Cloneable {//姓名private String name;//年龄private Integer age;/*注意:首先方法的权限修饰符需要更改为public方法的返回值可以更改为当前类的类名*/@Overridepublic Object clone() throws CloneNotSupportedException {return super.clone();}
}}    
/*** 案例:已知A对象的姓名为豹子头林冲,年龄30 。由于项目特殊要求需要将该对象的数据* 复制另外一个对象B中,并且此后A和B两个对象的数据不会相互影响** 浅拷贝的局限性*/
@SuppressWarnings("all")
public class CloneTest01 {public static void main(String[] args) throws Exception{//传统方式://创建学生对象Student stu1 = new Student("豹子头林冲",30);//再次创建一个新的学生对象Object stu2 = stu1.clone();System.out.println(stu1==stu2);System.out.println(stu1);System.out.println(stu2);System.out.println("----此时不管修改哪个对象的内容,stu1和stu2都不会受到影响----");stu1.setAge(33);System.out.println(stu1);System.out.println(stu2);}
}

运行程序

false
Student{name='豹子头林冲', age=30}
Student{name='豹子头林冲', age=30}
----此时不管修改哪个对象的内容,stu1和stu2都不会受到影响----
Student{name='豹子头林冲', age=33}
Student{name='豹子头林冲', age=30}

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;}@Overridepublic String toString() {return "Skill{" +"skillName='" + skillName + '\'' +'}';}@Overrideprotected Object clone() throws CloneNotSupportedException {return super.clone();}
}

学生类

public class Student  implements Cloneable {//姓名private String name;//年龄private Integer age;//技能private Skill skill;public Student(String name, int age, Skill skill) {this.name = name;this.age = age;this.skill = skill;}public Student() {}public Student(String name, Integer age) {this.name = name;this.age = age;}/*注意:首先方法的权限修饰符需要更改为public方法的返回值可以更改为当前类的类名*///浅克隆@Overridepublic Object clone() throws CloneNotSupportedException {return super.clone();}@Overridepublic String toString() {return "Student{" +"name='" + name + '\'' +", age=" + age +", 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;}
}

2.定义测试类
测试类

/*** 案例:已知A对象的姓名为豹子头林冲,年龄30 。由于项目特殊要求需要将该对象的数据* 复制另外一个对象B中,并且此后A和B两个对象的数据不会相互影响** 浅拷贝的局限性*/
@SuppressWarnings("all")
public class CloneTest02 {public static void main(String[] args) throws CloneNotSupportedException {//创建技能对象Skill s = new Skill("倒拔垂杨柳");//创建学生对象Student stu1 = new Student("鲁智深",30,s);//调用clone的方法进行数据的拷贝Object stu2 = stu1.clone();System.out.println(stu1 == stu2);System.out.println(stu1);System.out.println(stu2);System.out.println("----此时不管修改哪个对象的内容,stu1和stu2都不会受到影响----");//更改stu1对象的内容stu1.setAge(33);s.setSkillName("拳打镇关西");System.out.println(stu1);System.out.println(stu2);}
}

控制台效果

false
Student{name='鲁智深', age=30, skill=Skill{skillName='倒拔垂杨柳'}}
Student{name='鲁智深', age=30, skill=Skill{skillName='倒拔垂杨柳'}}
----此时不管修改哪个对象的内容,stu1和stu2都不会受到影响----
Student{name='鲁智深', age=33, skill=Skill{skillName='拳打镇关西'}}
Student{name='鲁智深', age=30, skill=Skill{skillName='拳打镇关西'}}

存在的问题: 基本数据类型可以达到完全复制,引用数据类型则不可以
原因: 在学生对象s被克隆的时候,其属性skill(引用数据类型)仅仅是拷贝了一份引用,因此当skill的值发生改
变时,被克隆对象s的属性skill也将跟随改变
深克隆
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;}@Overridepublic String toString() {return "Skill{" +"skillName='" + skillName + '\'' +'}';}@Overridepublic Skill clone() throws CloneNotSupportedException {return (Skill) super.clone();}
}

学生类

public class Student  implements Cloneable {//姓名private String name;//年龄private Integer age;//技能private Skill skill;public Student(String name, int age, Skill skill) {this.name = name;this.age = age;this.skill = skill;}public Student() {}public Student(String name, Integer age) {this.name = name;this.age = age;}public Skill getSkill() {return skill;}public void setSkill(Skill skill) {this.skill = skill;}/*注意:首先方法的权限修饰符需要更改为public方法的返回值可以更改为当前类的类名*/@Overridepublic Object clone() throws CloneNotSupportedException {//return super.clone();     //深拷贝,不能简单的调用父类的方法//先克隆出来一个学生对象Student stu = (Student) super.clone();//调用Skill类中的克隆方法,克隆出来一个Skill对象Skill skill = (Skill) this.skill.clone();//将克隆出来的skill赋值给stu该对象的成员变量stu.setSkill(skill);//需要把stu返回return stu;}@Overridepublic String toString() {return "Student{" +"name='" + name + '\'' +", age=" + age +", 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;}
}

2.定义测试类
测试类

/*** 案例:已知A对象的姓名为豹子头林冲,年龄30 。由于项目特殊要求需要将该对象的数据* 复制另外一个对象B中,并且此后A和B两个对象的数据不会相互影响** 深拷贝*/
@SuppressWarnings("all")
public class CloneTest02 {public static void main(String[] args) throws CloneNotSupportedException {//创建技能对象Skill s = new Skill("倒拔垂杨柳");//创建学生对象Student stu1 = new Student("鲁智深",30,s);//调用clone的方法进行数据的拷贝Object stu2 = stu1.clone();System.out.println(stu1 == stu2);System.out.println(stu1);System.out.println(stu2);System.out.println("----此时不管修改哪个对象的内容,stu1和stu2都不会受到影响----");//更改stu1对象的内容stu1.setAge(33);s.setSkillName("拳打镇关西");System.out.println(stu1);System.out.println(stu2);}
}

2.3 RandomAccess标记接口

介绍 标记接口由 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();
  1. 案例演示1
public class Test02 {public static void main(String[] args) {
//创建ArrayList集合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));System.out.println("----通过迭代器(顺序访问:)----");startTime = System.currentTimeMillis();Iterator<String> it = list.iterator();while (it.hasNext()){
//仅仅为了演示取出数据的时间,因此不对取出的数据进行打印it.next();}endTime = System.currentTimeMillis();System.out.println("顺序访问: "+(endTime-startTime));}
}

控制台效果

----通过索引(随机访问:)----
随机访问: 4
----通过迭代器(顺序访问:)----
顺序访问: 7

案例演示二

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));System.out.println("----通过迭代器(顺序访问:)----");startTime = System.currentTimeMillis();Iterator<String> it = list.iterator();while (it.hasNext()){
//仅仅为了演示取出数据的时间,因此不对取出的数据进行打印it.next();}endTime = System.currentTimeMillis();System.out.println("顺序访问: "+(endTime-startTime));}}

控制台效果

----通过索引(随机访问:)----
随机访问: 19535
----通过迭代器(顺序访问:)----
顺序访问: 5

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

小结: 由于随机访问的时候源码底层每次都需要进行折半的动作,再经过判断是从头还是从尾部一个个寻找。
而顺序访问只会在获取迭代器的时候进行一次折半的动作,以后每次都是在上一次的基础上获取下一个元素。
因此顺序访问要比随机访问快得多

实际开发应用场景

/*** 专门操作数据库*/
public class EmpDao {//创建一个JdbcTemplate对象private static JdbcTemplate jt = new JdbcTemplate(JdbcUtils.getDataSource());//查询所有@Testpublic void query(){//拼写SQLString sql = "select * from user";List<User> list = jt.query(sql, new BeanPropertyRowMapper<User>(User.class));/*建议进行判断判断查询返回的结果是否实现RandomAccess该接口如果实现,那么就推荐使用 随机遍历的方式迭代集合否则,就推荐使用顺序的方式迭代集合*///判断if(list instanceof RandomAccess) {//随机访问for (int i = 0; i < list.size(); i++) {User user = list.get(i);System.out.println(user);}}else{//顺序访问for (User user : list) {System.out.println(user);}}}

2.4 AbstractList抽象类

3. ArrayList源码分析

3.1 构造方法

在这里插入图片描述

3.2 案例演示

案例一:

空参构造ArrayList()

public class Test03 {public static void main(String[] args) {//这行代码做了什么?//真的构造一个初始容量为十的空列表?ArrayList<String> list = new ArrayList<String>();}
}

源码分析

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;}
}

结论
通过空参构造方法创建集合对象并未构造一个初始容量为十的空列表,仅仅 DEFAULTCAPACITY_EMPTY_ELEMENTDATA 的地址赋值给 elementData

案例二:

指定容量ArrayList(int initialCapacity)

public class Test03 {public static void main(String[] args) {
//这行代码ArrayList底层做了什么?ArrayList<String> list = new ArrayList<String>(5);}
}

源码分析

public class ArrayList<E> {public ArrayList(int initialCapacity) { //initialCapacity = 5
//判断初始容量initialCapacity是否大于0if (initialCapacity > 0) {
//创建一个数组,且指定长度为initialCapacitythis.elementData = new Object[initialCapacity];} else if (initialCapacity == 0) {
//如果initialCapacity容量为0,把EMPTY_ELEMENTDATA的地址赋值给elementDatathis.elementData = EMPTY_ELEMENTDATA;} else {
//以上两个条件都不满足报错throw new IllegalArgumentException("Illegal Capacity: "+initialCapacity);}}
}

结论

根据 ArrayList 构造方法参数创建指定长度的数组

案例三:

ArrayList(Collection<? extends E> c)

public class Test03 {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);}}
}

源码分析

public class ArrayList<E> {public ArrayList(Collection<? extends E> c) {// 将集合构造中的集合对象转成数组,且将数组的地址赋值给elementDataelementData = c.toArray();// 将elementData的长度赋值给 集合长度size,且判断是否不等于 0if ((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);}
}
//数组工具类
public 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) {//用三元运算符进行判断,不管结果如何都是创建一个新数组T[] copy = ((Object)newType == (Object)Object[].class)? (T[]) new Object[newLength]: (T[]) Array.newInstance(newType.getComponentType(), newLength);//将数组的内容拷贝到 copy 该数组中System.arraycopy(original, 0, copy, 0,Math.min(original.length, newLength));//返回拷贝元素成功后的数组return copy;}
}

3.3 添加方法

这个方法是直接往list里面添加元素,可以看到它会判断数组是否初始化了,如果没有这个时候才会真正的初始化数组,然后直接把元素放到数组中。
在这里插入图片描述
public boolean add(E e) 添加单个元素

将指定的元素添加到此列表的尾部。

public class Test01 {public static void main(String[] args) {ArrayList<String> list = new ArrayList<>();list.add("马伟奇");}
}

源代码

public class ArrayList<E> {//将添加的数据传入给 epublic boolean add(E e) {// 1. 检测是否需要扩容ensureCapacityInternal(size + 1);// 2. 将新元素插入序列尾部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++;//判断最小容量 - 数组长度是否大于 0if (minCapacity - elementData.length > 0)//将第一次计算出来的容量传递给 核心扩容方法grow(minCapacity);}private void grow(int minCapacity) {//记录数组的实际长度,此时由于木有存储元素,长度为0int oldCapacity = elementData.length;//核心扩容算法 原容量的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);// 调用数组工具类方法,创建一个新数组,将新数组的地址赋值给elementDataelementData = Arrays.copyOf(elementData, newCapacity);}
}

对于在元素序列尾部插入,这种情况比较简单,只需两个步骤即可:

  • 检测数组是否有足够的空间插入
  • 将新元素插入至序列尾部

在这里插入图片描述

public void add(int index, E element) 在指定索引处添加元素

这个方法和无参的add()方法的区别就是不是在数组后面顺序添加,而是指定位置的插入,可以看到检查是否扩容的逻辑没有变,主要是多了一步copy数组的操作,比如一个数组是[1,2,3,4,5],需要在索引1的位置插入一个10,会先从索引1的位置copy数组,把后面的数字都往后挪一位[1,2,2,3,4,5],然后再把索引1的2改为10,就成了[1,10,2,3,4,5],这个操作的时间复杂度就是O(n)了,因为最坏的情况就是插入到索引0,整个数组的值都要挪动一位。

public class Test01 {public static void main(String[] args) {ArrayList<String> list = new ArrayList<>();list.add("马伟奇1");list.add("马伟奇2");list.add("马伟奇3");list.add(1, "奥特曼");System.out.println(list);}
}

源代码

public class ArrayList<E> {public void add(int index, E element) {// 检查索引越界rangeCheckForAdd(index);// 1. 检测是否需要扩容//调用方法检验是否要扩容,且让增量++ensureCapacityInternal(size + 1);// copy数组,在要插入的索引位置开始一直到最后一个值,整体往后挪一格,然后把新值填入要插入的索引中// 2. 将 index 及其之后的所有元素都向后移一位// arraycopy(被复制的数组, 从第几个元素开始, 复制到哪里, 从第几个元素开始粘贴, 复制的元素个数)System.arraycopy(elementData, index, elementData, index + 1,size - index);// 3. 将新元素插入至 index 处elementData[index] = element;size++;}private void rangeCheckForAdd(int index) {//超出指定范围就报错if (index > size || index < 0)throw new IndexOutOfBoundsException(outOfBoundsMsg(index));}// 检查容量是否足够的方法private void ensureCapacityInternal(int minCapacity) {// 如果数组是空的,创建一个长度10的数组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);}
}

如果是在元素序列指定位置(假设该位置合理)插入,则情况稍微复杂一点,需要三个步骤:

  • 检测数组是否有足够的空间
  • 将 index 及其之后的所有元素向后移一位
  • 将新元素插入至 index 处

在这里插入图片描述
从上图可以看出,将新元素插入至序列指定位置,需要先将该位置及其之后的元素都向后移动一位,为新元素腾出位置。这个操作的时间复杂度为O(N),频繁移动元素可能会导致效率问题,特别是集合中元素数量较多时。在日常开发中,若非所需,我们应当尽量避免在大集合中调用第二个插入方法。

public boolean addAll(Collection<? extends E> c) 将集合的所有元素一次性添加到集合

按照指定 collection 的迭代器所返回的元素顺序,将该 collection 中的所有元素添加到此列表的尾部。

public class Test01 {public static void main(String[] args) {ArrayList<String> list = new ArrayList<>();list.add("马伟奇1");list.add("马伟奇2");list.add("马伟奇3");ArrayList<String> list1 = new ArrayList<>();list1.addAll(list);System.out.println(list);System.out.println(list1);}
}
[马伟奇1, 马伟奇2, 马伟奇3]
[马伟奇1, 马伟奇2, 马伟奇3]

源码分析

public static void arraycopy(Object src, int srcPos, Object dest, int destPos, int length)。 它的根本目的就是进行数组元素的复制。即从指定源数组中复制一个数组,复制从指定的位置开始,到目标数组的指定位置结束。将源数组src从srcPos位置开始复制到dest数组中,复制长度为length,数据从dest的destPos位置开始粘贴。

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) 在指定的索引位置添加集合

追加集合的方法也可以指定索引,就是addAll(Collection<? extends E> c)和add(int index, E element)的结合版,计算出索引后面的值要移动的位置进行移动,再把追加的集合的值copy到指定索引,时间复杂度也是O(n),但是n是追加集合的长度+数组移动元素的长度。

public class Test01 {public static void main(String[] args) {ArrayList<String> list = new ArrayList<>();list.add("马伟奇1");list.add("马伟奇2");list.add("马伟奇3");ArrayList<String> list1 = new ArrayList<>();list1.add("奥特曼");list1.add("葫芦娃");
//在指定索引处添加一个集合list1.addAll(1,list);System.out.println(list);System.out.println(list1);}
}

源码分析

public class ArrayList<E> {//长度为0的空数组private static final Object[] EMPTY_ELEMENTDATA = {};//默认容量为空的数组private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {};//集合存元素的数组Object[] elementData;//集合的长度private int size;//默认的容量private static final int DEFAULT_CAPACITY = 10;public boolean addAll(int index, Collection<? extends E> c) {
//校验索引rangeCheckForAdd(index);
//将数据源转成数组Object[] a = c.toArray();
//记录数据源的长度 3int numNew = a.length;
//目的就是为了给集合存储数据的数组进行扩容ensureCapacityInternal(size + numNew);
//numMoved:代表要移动元素的个数 --> 1个
//numMoved: 数据目的(集合list1)的长度-调用addAll的第一个参数 (索引1)int numMoved = size - index;
//判断需要移动的个数是否大于0if (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 final class System {参数src - 源数组。srcPos - 源数组中的起始位置。dest - 目标数组。destPos - 目的地数据中的起始位置。length - 要复制的数组元素的数量。public static void arraycopy(Object src,int srcPos,Object dest,int destPos,int length)
}

如何计算元素移动的位置&数量

public class Test01 {public static void main(String[] args) {//数据源: listString[] a = {"马伟奇1","马伟奇2","马伟奇3"};//数据目的: list1String[] arr = {"奥特曼","葫芦娃",null,null,null,null,null,null,null,null};/*int numNew = a.length;int numMoved = size - index;if (numMoved > 0)System.arraycopy(elementData, index, elementData, index + numNew,numMoved);*///获取数据源的长度 3int numNew = a.length;//numMoved = 集合真实长度 - 要存的索引位置//要移动元素的个数为:1int numMoved = 2 - 1;//判断是否需要移动元素if (numMoved > 0)//src - 源数组。//srcPos - 源数组中的起始位置。//dest - 目标数组。//destPos - 目的地数据中的起始位置。//length - 要复制的数组元素的数量System.arraycopy(arr, 1, arr, 4, numMoved);System.out.println(Arrays.toString(arr));}
}

运行程序

[奥特曼, 葫芦娃, null, null, 葫芦娃, null, null, null, null, null]

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对应的元素赋值给 oldValueE oldValue = elementData(index);//计算集合需要移动元素个数int numMoved = size - index - 1;//如果需要移动元素个数大于0,就使用arrayCopy方法进行拷贝//注意:数据源和数据目的就是elementDataif (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) {//判断要删除的元素是否为nullif (o == null) {//遍历集合for (int index = 0; index < size; index++)//判断集合的元素是否为nullif (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该元素,那么就会返回falsereturn 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);}
}

源码分析

public class ArrayList<E> {public E set(int index, E element) {//范围校验rangeCheck(index);//先取出index对应的元素,且赋值给oldValueE 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();}}
}
//ArrayList亲爹
public abstract class AbstractList<E> extends AbstractCollection<E> {}
//ArrayList亲爷爷
public abstract class AbstractCollection<E> {public String toString() {
//注意:此时相当于用ArrayList对象在调用iterator()方法 获取迭代器
//那么这个时候需要先看看ArrayList中的iterator()方法Iterator<E> it = iterator();
//调用ArrayList中hasNext方法判断是否有元素,如果hasNext()方法返回false
//那么就toString方法就返回一个 "[]"if (! it.hasNext())return "[]";
//创建StringBuilder,对集合的内容进行拼接,避免字符串频繁拼接产生很多无效对象StringBuilder sb = new StringBuilder();sb.append('[');
//无限循环for (;;) {
//调用ArrayList中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”,使用迭代器遍
历获取集合的每一个元素

//案例一: 已知集合:List<String> list = new ArrayList<String>();里面有三个元素:"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.ConcurrentModificationExceptionat java.util.ArrayList$Itr.checkForComodification(ArrayList.java:911)at java.util.ArrayList$Itr.next(ArrayList.java:861)at com.maweiqi.Test01.main(Test01.java:26)

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

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的值也是 3int expectedModCount = modCount;public boolean hasNext() {return cursor != size;}//获取元素的方法public E next() {//每次获取元素,会先调用该方法校验 预期修改次数是否 == 实际修改次数/*tips:if(s.equals("hello")) {list.remove("hello");}当if表达式的结果为true,那么集合就会调用remove方法*/checkForComodification();//把下一个元素的索引赋值给iint i = cursor;//判断是否有元素if (i >= size)throw new NoSuchElementException();//将集合底层存储数据的数组赋值给迭代器的局部变量 elementDataObject[] elementData = ArrayList.this.elementData;//再次判断,如果下一个元素的索引大于集合底层存储元素的长度 并发修改异常//注意,尽管会产生并发修改异常,但是这里显示不是我们要的结果if (i >= elementData.length)throw new ConcurrentModificationException();//每次成功获取到元素,下一个元素的索引都是当前索引+1cursor = i + 1;//返回元素return (E) elementData[lastRet = i];}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;}
}

案例三:已知集合: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]

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);}
}

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

public class ArrayList<E> {public Iterator<E> iterator() {return new Itr();}//ArrayList内部类
//一定要注意观察 Itr 类中的几个成员变量//ArrayList内部类
//一定要注意观察 Itr 类中的几个成员变量private class Itr implements Iterator<E> {int cursor; // 下一个要返回元素的索引int lastRet = -1; // 最后一个返回元素的索引//将实际修改集合次数 赋值 给预期修改次数
//在迭代的过程中,只要实际修改次数和预期修改次数不一致就会产生并发修改异常
//由于expectedModCount是Itr的成员变量,那么只会被赋值一次!!!
//同时由于集合调用了三次add方法,那么实际修改集合次数就是 3,因此expectedModCount的值也是 3int expectedModCount = modCount;public boolean hasNext() {return cursor != size;}//获取元素的方法public E next() {checkForComodification();
//把下一个元素的索引赋值给iint i = cursor;
//判断是否有元素if (i >= size)throw new NoSuchElementException();
//将集合底层存储数据的数组赋值给迭代器的局部变量 elementDataObject[] elementData = ArrayList.this.elementData;
//再次判断,如果下一个元素的索引大于集合底层存储元素的长度 并发修改异常
//注意,尽管会产生并发修改异常,但是这里显示不是我们要的结果if (i >= elementData.length)throw new ConcurrentModificationException();
//每次成功获取到元素,下一个元素的索引都是当前索引+1cursor = 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赋值给cursorcursor = lastRet;
//再次等于-1lastRet = -1;
//再次将集合实际修改次数赋值给预期修改次数,那么这个时候不管集合自身是否删除成功
//那么实际修改次数和预期修改次数又一致了,所以并不会产生并发修改异常expectedModCount = modCount;} catch (IndexOutOfBoundsException ex) {throw new ConcurrentModificationException();}}//快速删除方法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);}
}

源码分析

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

4.0 包含方法

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);}
}

源码分析

public class ArrayList<E> {//源码contains方法public boolean contains(Object o) {
//调用indexOf方法进行查找return indexOf(o) >= 0;}public int indexOf(Object o) {
//如果元素是null,也进行遍历操作
//因为集合中有可能够会存储nullif (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;}
}

结论:底层也是通过循环遍历集合,取出一个个的元素和要找的元素进行比较

4.1 判断集合是否为空

public boolean isEmpty()

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);}
}

源码分析

public class ArrayList<E> {public boolean isEmpty() {return size == 0;}
}
  1. 面试题
    4.1 ArrayList是如何扩容的?
    源码分析过程中已经讲解
    第一次扩容10
    以后每次都是原容量的1.5倍
    public class Test01 {
    public static void main(String[] args) {
    //创建集合对象
    List list = new ArrayList();
    //添加元素
    list.add(“hello”);
    list.add(“PHP”);
    list.add(“Java”);
    boolean b = list.isEmpty();
    System.out.println(b);
    System.out.println(list);
    }
    }
    public class ArrayList {
    public boolean isEmpty() {
    return size == 0;
    }
    }

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();
//需求:还需要添加10W条数据for (int i = 0; i < 100000; i++) {list.add(i+"");}long endTime = System.currentTimeMillis();System.out.println("花费时间: "+ (endTime - startTime));}
}

解决方案

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();
//需求:还需要添加10W条数据for (int i = 0; i < 100000; i++) {list.add(i+"");}long endTime = System.currentTimeMillis();System.out.println("花费时间: "+ (endTime - startTime));//创建集合的时候指定足够大的容量List<String> list1 = new ArrayList<String>(100000);startTime = System.currentTimeMillis();for (int i = 0; i < 100000; i++) {list1.add(i+"");}endTime = System.currentTimeMillis();System.out.println("花费时间: "+ (endTime - startTime));}
}

注意:这种优化方式只针对特定的场景,如果添加的元素是少量的、未知的,不推荐使用

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

根据索引删除
案例:ArrayList和LinkedList对比

public class Test01 {public static void main(String[] args) {
//创建ArrayList集合对象ArrayList<String> arrayList = new ArrayList<String>();
//添加500W个元素for (int i = 0; i < 5000000; i++) {arrayList.add(i+"马伟奇");}
//获取开始时间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));
//创建LinkedList集合对象LinkedList<String> linkedList = new LinkedList<String>();
//添加500W个元素for (int i = 0; i < 5000000; i++) {linkedList.add(i+"马伟奇");}
//获取开始时间startTime = System.currentTimeMillis();
//根据索引删除LinkedList集合元素
//删除索引5000对应的元素value = arrayList.remove(50000);System.out.println(value);endTime = System.currentTimeMillis();System.out.println("LinkedList集合删除元素的时间: "+(endTime-startTime));}
}

输出

50000马伟奇
ArrayList集合删除元素的时间: 4
50001马伟奇
LinkedList集合删除元素的时间: 3

源码分析

ArrayList根据索引删除元素源码

public class ArrayList<E> {public E remove(int index) {
//范围校验rangeCheck(index);
//增量++modCount++;
//将index对应的元素赋值给 oldValueE oldValue = elementData(index);
//计算集合需要移动元素个数int numMoved = size - index - 1;
//如果需要移动元素个数大于0,就使用arrayCopy方法进行拷贝
//注意:数据源和数据目的就是elementDataif (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;
//被删除元素的上一个节点置为nullx.prev = null;}
//如果被删除元素的下一个节点为null,最后一个节点就等于被删除元素的上一个节点if (next == null) {last = prev;} else {
//否则,被删除节点的下一个节点 等于被删除节点的前一个节点next.prev = prev;
//被删除元素的下一个节点置为nullx.next = null;}
//被删除元素的内容置为nullx.item = null;
//集合长度--size--;
//实际修改次数++modCount++;
//返回被删除的元素return element;}
}

结论

    1. 数组删除元素确实要比链表慢,慢在需要创建新数组,还有比较麻烦的数据拷贝,但是在ArrayList
      底层不是每次删除元素都需要扩容,因此在这个方面相对于链表来说数组的性能更好
    1. LinkedList删除元素之所以效率并不高,其原理在于底层先需要对整个集合进行折半的动作,然后
      又需要对集合进行遍历一次,这些操作导致效率变低

根据元素删除

案例:ArrayList和LinkedList对比

public class Test01 {public static void main(String[] args) {
//创建ArrayList集合对象ArrayList<String> arrayList = new ArrayList<String>();
//添加500W个元素for (int i = 0; i < 5000000; i++) {arrayList.add(i+"XXX");}
//获取开始时间long startTime = System.currentTimeMillis();
//根据元素删除ArrayList集合元素
//删除元素为 "5000XXX"boolean b = arrayList.remove("5000XXX");System.out.println("删除的状态: "+b);
//获取结束时间long endTime = System.currentTimeMillis();System.out.println("ArrayList集合删除元素的时间: "+(endTime-startTime));
//创建LinkedList集合对象LinkedList<String> linkedList = new LinkedList<String>();
//添加500W个元素for (int i = 0; i < 5000000; i++) {linkedList.add(i+"XXX");}
//获取开始时间startTime = System.currentTimeMillis();
//根据元素删除LinkedList集合元素
//删除元素为 "5000XXX"b = linkedList.remove("5000XXX");System.out.println("删除的状态: "+b);endTime = System.currentTimeMillis();System.out.println("LinkedList集合删除元素的时间: "+(endTime-startTime));}
}

源码分析
ArrayList根据元素删除元素

public class ArrayList<E> {public boolean remove(Object o) {
//判断要删除的元素是否为nullif (o == null) {
//遍历集合for (int index = 0; index < size; index++)
//判断集合的元素是否为nullif (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该元素,那么就会返回falsereturn 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;}
}

LinkedList根据元素删除元素

public class LinkedList<E> {//LinkedList集合底层删除源码public boolean remove(Object o) {
//判断要删除的元素是否为null
//不管是否为null都从第一个元素开始,从头部往后找
//找到之后,调用unlink方法进行解绑,更改节点和节点之间记录的地址if (o == null) {for (Node<E> x = first; x != null; x = x.next) {if (x.item == null) {unlink(x);return true;}}} else {for (Node<E> x = first; x != null; x = x.next) {if (o.equals(x.item)) {unlink(x);return true;}}}return false;}E unlink(Node<E> x) {final E element = x.item;final Node<E> next = x.next;final Node<E> prev = x.prev;if (prev == null) {first = next;} else {prev.next = next;x.prev = null;}if (next == null) {last = prev;} else {next.prev = prev;x.next = null;}x.item = null;size--;modCount++;return element;}
}

4.4 ArrayList是线程安全的么?

ArrayList不是线程安全的,使用一个案例演示

//线程任务类
public class CollectionTask implements Runnable {//通过构造方法共享一个集合private List<String> list;public CollectionTask(List<String> list) {this.list = list;}@Overridepublic void run() {try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}
//把当前线程名字加入到集合list.add(Thread.currentThread().getName());}
}//测试类
public 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());}
}

需要线程安全怎么办?

方式一:使用Collections.synchronizedList(list)

public class CollectionTask implements Runnable {//通过构造方法共享一个集合private List<String> list;public CollectionTask(List<String> list) {this.list = list;}@Overridepublic void run() {try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}
//把当前线程名字加入到集合list.add(Thread.currentThread().getName());}
}//测试类
public class CollectionTest01 {public static void main(String[] args) throws InterruptedException {
//创建集合List<String> list = new ArrayList<String>();
//通过Collections工具类把List变成一个线程安全的集合list = Collections.synchronizedList(list);
//创建线程任务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());}
}

方式二:使用

public class CollectionTask implements Runnable {//通过构造方法共享一个集合private List<String> list;public CollectionTask(List<String> list) {this.list = list;}@Overridepublic void run() {try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}
//把当前线程名字加入到集合list.add(Thread.currentThread().getName());}
}//测试类
public class CollectionTest01 {public static void main(String[] args) throws InterruptedException {
//创建线程安全的集合类VectorList<String> list = new Vector<>();
//通过Collections工具类把List变成一个线程安全的集合list = Collections.synchronizedList(list);
//创建线程任务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());}
}

实际开发场景

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

//线程任务类
public class Test01 {//创建JdbcTemplate对象JdbcTemplate jt = new JdbcTemplate(JDBCUtils.getDataSource());//3.利用JDBC查询出基础班在读的男学员的所有信息按成绩的降序输出到控制台上(利用JDBC)@Testpublic void fun1() throws Exception {
//拼写SQLString sql = "select * from stutb where sex = ? and type like ? order by scoredesc";
//调用方法查询 将结果集的每一行都封装成一个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");}@Overridepublic 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();}}
}

读写分离集合

class CollectionThread implements Runnable{private static CopyOnWriteArrayList<String> list = new CopyOnWriteArrayList<>();static{list.add("Jack");list.add("Lucy");list.add("Jimmy");}@Overridepublic 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();}}
}

4.7 ArrayList 和 LinkList区别?

  • ArrayList

    • 基于动态数组的数据结构
    • 对于随机访问的get和set,ArrayList要优于LinkedList
    • 对于随机操作的add和remove,ArrayList不一定比LinkedList慢 (ArrayList底层由于是动态数组,因此
      并不是每次add和remove的时候都需要创建新数组)
  • LinkedList

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/49521.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

如何将应用程序发布到 App Store

憧憬blog主页 在强者的眼中&#xff0c;没有最好&#xff0c;只有更好。我们是移动开发领域的优质创作者&#xff0c;同时也是阿里云专家博主。 ✨ 关注我们的主页&#xff0c;探索iOS开发的无限可能&#xff01; &#x1f525;我们与您分享最新的技术洞察和实战经验&#xff0…

伦敦金短线好还是长线好

在伦敦金投之中&#xff0c;长期有一个争论很久的问题&#xff0c;那就是伦敦金投资究竟是长线好还是短线好&#xff1f;不同的投资者对这个问题有不同的看法&#xff0c;一般认为&#xff0c;伦敦金投资比较适合短线交易。笔者也将讨论这个问题&#xff0c;看看伦敦金投资是不…

MinIO线上扩容实战

硬件投入肯定是随着业务的增长而增长&#xff0c;这就要求中间件平台必须提供水平伸缩机制&#xff0c;MinIO对象存储服务也不例外&#xff0c;本文就详细介绍MinIO的扩容。 Minio支持通过增加新的Server Pool来扩容老的集群。每个Server Pool都是一个相对独立的故障域&#x…

​深度解析温湿度监控系统环境对档案存储库房的影响

深度解析温湿度监控系统环境对档案存储库房的影响 &#xff08;一&#xff09;高温高湿环境对档案材料的影响 1、库房高温高湿会使一些档案纸张材料中耐热性较差的字迹产生油渗扩散褪变等现象而导致字迹模糊不清&#xff0c;会加速纸张中残留的有害化学物质对纤维素的破坏潮湿…

【C++ 学习 ⑭】- 详解 stack、queue 和 priority_queue 容器适配器

目录 一、详解 C STL 容器适配器 1.1 - 什么是容器适配器&#xff1f; 1.2 - 容器适配器的种类 二、详解 C STL deque 容器 2.1 - deque 的原理介绍 2.2 - deque 的优缺点 三、详解 stack 容器适配器 3.1 - stack 的基本介绍 3.2 - stack 的成员函数 3.3 - stack 的模…

androidstudio Please specify a signing configuration for this variant (release)

当直接运行release版本时&#xff0c;报错Error: The apk for your currently selected variant cannot be signed. Please specify a signing configuration for this variant (package64-release). 解决报错&#xff1a;添加签名&#xff0c;signingConfigs 写在buildTypes前…

java八股文面试[java基础]——final 关键字作用

为什么局部内部类和匿名内部类只能访问final变量&#xff1a; 知识来源 【基础】final_哔哩哔哩_bilibili

壁仞科技与百度飞桨完成II级兼容性测试

近日&#xff0c;壁仞科技BR104通用GPU与百度飞桨已完成II级兼容性测试。测试结果显示&#xff0c;双方兼容性表现良好&#xff0c;整体运行稳定。这是壁仞科技加入飞桨“硬件生态共创计划”后的阶段性成果。产品兼容性证明本次II级兼容性测试完成了涵盖自然语言处理、计算机视…

服务器的介绍

1.服务器概述 1.1 服务器的基本概念 服务器是计算机的一种&#xff0c;是网络中为客户端计算机提供各种服务的高性能计算机&#xff1b; 服务器在网络操作系统的控制下&#xff0c;将与其相连的硬盘、磁带、 打印机及昂贵的专用通讯设备提供给网络上的客户站点共享&#xf…

【Unity3D赛车游戏】【二】如何制作一个真实模拟的汽车

&#x1f468;‍&#x1f4bb;个人主页&#xff1a;元宇宙-秩沅 &#x1f468;‍&#x1f4bb; hallo 欢迎 点赞&#x1f44d; 收藏⭐ 留言&#x1f4dd; 加关注✅! &#x1f468;‍&#x1f4bb; 本文由 秩沅 原创 &#x1f468;‍&#x1f4bb; 收录于专栏&#xff1a;Uni…

C# 学习笔记

C# 学习笔记 Chapter 1 C# 基础部分Section 1 类与命名空间Part 1 命名空间 NameSpacePart 2 类 Class Section 2 基本元素Section 3 数据类型Part 1 什么是类型&#xff1f;Part 2 类型在 C Sharp 中的作用Part 3 C Sharp 中的数据类型 Section 4 变量、对象与内存Part 1 变量…

k8s扩缩容与滚动更新

使用kubectl run创建应用 kubectl run kubernetes-bootcamp \> --imagedocker.io/jocatalin/kubernetes-bootcamp:v1 \> --port8080 端口暴露出去 kubectl expose pod kubernetes-bootcamp --type"NodePort" --port 8080 使用kubectl create创建应用 kubect…

stack和queue的模拟实现

stack和queue的模拟实现 容器适配器什么是适配器STL标准库中stack和queue的底层结构deque的简单介绍deque的缺陷 stack模拟实现queue模拟实现priority_queuepriority_queue的使用priority_queue的模拟实现 容器适配器 什么是适配器 适配器是一种设计模式(设计模式是一套被反复…

【C++】红黑树

目录 一、红黑树的概念二、红黑树的性质三、红黑树的插入操作四、红黑树的验证五、红黑树和AVL树的比较六、代码 一、红黑树的概念 红黑树&#xff0c;是一种二叉搜索树&#xff0c;但在每个结点上增加一个存储位表示结点的颜色&#xff0c;可以是Red或Black。 通过对任何一条从…

设计模式(9)建造者模式

一、 1、概念&#xff1a;将一个复杂对象的构造与它的表示分离&#xff0c;使得同样的构造过程可以创建不同的表示。建造者模式主要用于创建一些复杂的对象&#xff0c;这些对象内部构建间的顺序通常是稳定的&#xff0c;但对象内部的构建通常面临着复杂的变化&#xff1b;建造…

[SpringBoot3]Web服务

五、Web服务 基于浏览器的B/S结构应用十分流行。SpringBoot非常适合Web应用开发&#xff0c;可以使用嵌入式Tomcat、Jetty、Undertow或Netty创建一个自包含的HTTP服务器。一个SpringBoot的Web应用能够自己独立运行&#xff0c;不依赖需要安装的Tomcat、Jetty等。SpringBoot可以…

indexDB入门到精通

前言 由于开发3D可视化项目经常用到模型&#xff0c;而一个模型通常是几m甚至是几十m的大小对于一般的服务器来讲加载速度真的十分的慢&#xff0c;为了解决这个加载速度的问题&#xff0c;我想到了几个本地存储的。 首先是cookie,cookie肯定是不行的&#xff0c;因为最多以只…

Vue的Ajax请求-axios、前后端分离练习

Vue的Ajax请求 axios简介 ​ Axios&#xff0c;是Web数据交互方式&#xff0c;是一个基于promise [5]的网络请求库&#xff0c;作用于node.js和浏览器中&#xff0c;它是 isomorphic 的(即同一套代码可以运行在浏览器和node.js中)。在服务端它使用原生node.js http模块, 而在…

SpringBoot +Vue3 简单的前后端交互

前端&#xff1a;Vue3 创建项目&#xff1a; npm create vuelatest > cd <your-project-name> > npm install > npm run dev 项目结构图如下&#xff1a; 1、查看入口文件内容&#xff1a;main.js 代码如下&#xff1a; import ./assets/main.css impor…

自己实现 SpringMVC 底层机制 系列之-实现任务阶段 6-完成控制器方法获取参数-@RequestParam

&#x1f600;前言 自己实现 SpringMVC 底层机制 系列之-实现任务阶段 6-完成控制器方法获取参数-RequestParam &#x1f3e0;个人主页&#xff1a;尘觉主页 &#x1f9d1;个人简介&#xff1a;大家好&#xff0c;我是尘觉&#xff0c;希望我的文章可以帮助到大家&#xff0c…