Java基础16(集合框架 List ArrayList容器类 ArrayList底层源码解析及扩容机制)

news2024/12/22 19:50:17

目录

一、什么是集合?

二、集合接口

三、List集合

四、ArrayList容器类

1. 常用方法

1.1 增加

1.2 查找

int  size() 

E  get(int index) 

int indexOf(Object  c) 

boolean contains(Object  c)

boolean  isEmpty()

List  SubList(int fromindex,int  toIndex)

boolean equals(Object o)

1.3 修改

E set(int index, E element)

1.1.4 删除

 boolean removeAll(Collection c)--- 差集

boolean retainAll(Collection c)   --交集 

1.5 其他方法

clone()  

sort()

List和Array转换

2. 遍历

2.1 for循环

2.2 增强for循环

2.3 迭代器

Iterator的remove 

3. 初始化

无参初始化

有参初始化

五、ArrayList底层源码解析及扩容机制(⭐)

【特点】 

【数据结构】

扩容机制:

 总结:

小结


一、什么是集合?

        什么是集合?集合就是“由若干个确定的元素所构成的整体”,在程序中,一般代表保存若干个元素(数据)的某种容器类。在数学中,我们经常遇到集合的概念。例如:


●有限集合:

  • 一个班所有的同学构成的集合;
  • 一个网站所有的商品构成的集合;
  • ...

●无限集合:

  • 全体自然数集合:1,2,3,……
  • 有理数集合;
  • 实数集合;
  • ...


为什么要在计算机中引入集合呢?这是为了便于处理一组类似的数据,例如:

  • 计算所有同学的总成绩和平均成绩;
  • 列举所有的商品名称和价格;
  • ……


        在Java中,如果一个Java对象可以在内部持有(保存)若干其他Java对象,并对外提供访问接口,我们把这种Java对象的容器称为集合。很显然,Java的数组也可以看作是一种集合:

        既然Java提供了数组这种数据类型,可以充当集合,那么,我们为什么还需要其他集合类?这是因为数组有如下限制:

  • 长度开始时必须指定,且一旦指定,不能更改;即数组长度很难扩容
  • 保存的必须为同一类型的元素;
  • 数组的方法和功能不完善,查看数组中的具体长度,只有length属性知道内存大小,不知道具体的元素个数。 

public static void main(String[] args) {
		String[] menu = new String[10];
		menu[0] = "麻婆豆腐";
		menu[1] = "麻辣烫";
		menu[2] = "null";
		System.out.println(menu.length);//10

		ArrayList<String> arrayList = new ArrayList<String>();
		for (int i = 0; i < 3; i++) {
			arrayList.add("麻婆豆腐");
			arrayList.add("麻辣烫");
			arrayList.add("null");
//			arrayList.add( 12 );
//			arrayList.add( 12.23 );
//			arrayList.add( true );
//			arrayList.add(new Date());
		}
		System.out.println(arrayList);//[麻婆豆腐, 麻辣烫, null, 麻婆豆腐, 麻辣烫, null, 麻婆豆腐, 麻辣烫, null]
		System.out.println(arrayList.size());//9
	}


因此,我们需要各种不同类型的集合类来处理不同的数据,例如:

  • 可变大小的顺序链表;
  • 保证无重复元素的集合;
  • ...

二、集合接口

        Java标准库自带的java.util包提供了集合相关的接口和实现类:Collection接口,它是除Map接口外所有其他集合类的根接口。

Collection接口:collection接口是单例集合的跟接口,英文含义集中、收集

单列集合是将数据进行一个一个的存储,存储的是值

Java的java.util包主要提供了以下三种类型的集合:

  • List:一种有序列表的集合;
  • Set:一种保证没有重复元素的集合;
  • Map:一种通过键值(key-value)查找的映射表集合,例如,根据Student的name查找对应Student的Map。

        Java集合的设计有几个特点:首先实现了接口和实现类相分离,例如,有序表的接口是List,具体的实现类有ArrayList,LinkedList等;其次支持泛型,我们可以限制在一个集合中只能放入同一种数据类型的元素,例如:

List<String> list = new ArrayList<>(); // 只能放入String类型

        最后,Java访问集合总是通过统一的方式——迭代器(Iterator)来实现,它最明显的好处在于无需知道集合内部元素是按什么方式存储的。
        另外,由于Java的集合设计非常久远,中间经历过大规模改进,我们要注意到有一小部分集合类是遗留类,不应该继续使用:

  • Hashtable:一种线程安全的Map实现;
  • Vector:一种线程安全的List实现;
  • Stack:基于Vector实现的LIFO的栈;

三、List集合

        在集合类中,List是最基础的一种集合:它是一种有序列表。List的行为和数组几乎完全相同:List内部按照放入元素的先后顺序存放,每个元素都可以通过索引确定自己的位置,List的索引和数组一样,从0开始。

List的特点:

  • 元素有序的-即存入顺序和取出顺序是一致的(注意区分有序和排序)
  • 是可以重复的

我们观察List<E>接口,可以看到几个主要的接口方法:

  • 在末尾添加一个元素:boolean add(E e)
  • 在指定索引添加一个元素:boolean add(int index, E e)
  • 删除指定索引的元素:E remove(int index)
  • 删除某个元素:boolean remove(Object e)
  • 获取指定索引的元素:E get(int index)
  • 获取链表大小(包含元素的个数):int size()

四、ArrayList容器类

1. 常用方法

方法名说明
public boolean add(E e)将指定的参数元素追加到集合的末尾
public void add(int index ,E e)在集合的指定位置添加指定的元素(插入元素)
public void addAll(E object)用于将指定集合中所有元素添加到当前集合中
boolean addAll(int  index,Collection<? extends E> c)  从指定的位置开始,将指定 collection 中的所有元素插入到此列表中
方法名说明
public boolean remove(Object o)删除指定的元素,成功则返回true
public E remove(int index)删除指定索引位置的元素,返回被删除的元素
public E set(int index,E e)修改指定索引位置的元素,返回修改前的元素
public E get(int index)获取指定索引对应的元素
public int size()获取结合中元素个数

1.1 增加

  • boolean add(E,e)添加指定元素到集合尾部
    //泛型使用的目的:帮助我们建立类型安全的集合,添加元素的时候传入实际的类型
    		//泛型使用好处: 代码可读性增强,程序更安全
    		//泛型集合--泛型必须是引用数据类型
    		ArrayList<String> arrayList =new ArrayList<String>();
    
    //add(元素)添加元素到集合的尾部
    		boolean b = arrayList.add("z张三");
    		System.out.println("添加z张三是否成功:"+b);
    		arrayList.add("l李四");
    		arrayList.add("w王五");
    		arrayList.add("r任六");
    		arrayList.add("z张三");
    		System.out.println(arrayList);
    
    运行结果:
    添加z张三是否成功:true
    [z张三, l李四, w王五, r任六, z张三]
  •  void add(int index, E element)添加新元素到集合指定的下标位置
    //add(int index, E element)添加新元素到集合指定的下标位置
    	    arrayList.add(0,"孙二娘");
    	    arrayList.add(3,"刘备");
    	    System.out.println(arrayList);//[孙二娘, z张三, l李四, 刘备, w王五, r任六, z张三]
    
  • boolean addAll(Collection<? extends E> c)  添加集合C 内所有元素到当前集合 
    //使用Arrays.asList方法快速生成一个List集合
    	    //Arrays.asList返回的不是java.util下的ArrayList
    	    List<String> list = Arrays.asList("李白","高适","李清照","杜甫");
    	    
    	    
    	    //boolean addAll(Collection<? extends E> c)  添加集合C 内所有元素到当前集合
    	    arrayList.addAll(list);
    	    System.out.println(arrayList);//[孙二娘, z张三, l李四, 刘备, w王五, r任六, z张三, 李白, 高适, 李清照, 杜甫]
  • boolean addAll(int  index,Collection<? extends E> c)  从指定的位置开始,将指定 collection 中的所有元素插入到此列表中
    //boolean addAll(Collection<? extends E> c)  添加集合C 内所有元素到当前集合
    	    arrayList.addAll(list);
    	    System.out.println(arrayList);//[李白, 高适, 李清照, 杜甫, 孙二娘, z张三, l李四, 刘备, w王五, r任六, z张三, 李白, 高适, 李清照, 杜甫]
    

1.2 查找

int  size() 

查集合的长度,具体的元素个数

ArrayList<String> arrayList = new ArrayList<String>();
		arrayList.addAll(Arrays.asList("李白","高适","李清照","杜甫"));
		//1.获取元素的个数
		int size = arrayList.size();
		System.out.println("元素的个数为:"+size);//元素的个数为:4
E  get(int index) 

获取下标中的元素

//2.E get(int index)获取下标中的元素,
		System.out.println("首元素:"+arrayList.get(0));//首元素:李白
		System.out.println("尾元素:"+arrayList.get(arrayList.size()-1));//尾元素:杜甫
int indexOf(Object  c) 

查找找到指定元素的下标位置,如果不存在,则返回数组-1 

//3.int indexOf(Object  c) 查找找到指定元素的下标位置,如果不存在,则返回数组-1
		int index =arrayList.indexOf("高适1");
		System.out.println("高适所对应的下标位置:"+index);//高适所对应的下标位置:-1
boolean contains(Object  c)

判断集合中是否存在元素

//4.boolean contains(Object  c)
		System.out.println("高适是否存在:"+arrayList.contains("高适1"));//高适是否存在:false
boolean  isEmpty()

判断集合是否为空

 //5.判断集合是否为空boolean  isEmpty()
		System.out.println("集合是否为空:"+arrayList.isEmpty());//集合是否为空:false
List<E>  SubList(int fromindex,int  toIndex)

        subList是List接口中定义的一个方法,该方法主要用于返回一个集合中的一段、可以理解为截取一个集合中的部分元素,他的返回值也是一个List。

		//6.List<E>  SubList(int fromindex,int  toIndex)
	    List<String> subList = arrayList.subList(0, 3);
	    System.out.println(subList);//[李白, 高适, 李清照]
boolean equals(Object o)

  ArrayList.equals(Object obj) 方法比较的是两个列表(ArrayList或其它类型的List)的内容是否相等,即两个列表的大小必须相同,并且在相同索引位置上的元素也必须相等。

    //7.equals()
	    boolean bequals = arrayList.equals(subList);
	    System.out.println(bequals);//false

1.3 修改

E set(int index, E element)

修改指定索引位置的元素,返回修改前的元素

ArrayList<String> arrayList = new ArrayList<String>();
		arrayList.addAll(Arrays.asList("李白", "高适", "李清照", "杜甫"));
		System.out.println("原集合的内容:" + arrayList);

		// 修改指定下标的元素
		String result = arrayList.set(0, "王维");
		System.out.println("修改的元素:" + result);
		System.out.println("修改后的集合:" + arrayList);

1.1.4 删除

  • E remove(int index):根据指定索引删除元素,并把删除的元素返回 如果下标超过集合的最大下标,输出IndexOutOfBoundsException
// E remove(int index):根据指定索引删除元素,并把删除的元素返回
		String oldValue = arrayList.remove(0);
		System.out.println("删除的下标为0的元素为:" + oldValue);//删除的下标为0的元素为:李白
  • boolean remove(Object o):从集合中删除指定的元素,删除一个就返回 如果查到或者删除1个就返回,并不是删除所有相同的集合元素
//boolean remove(Object o):从集合中删除指定的元素,删除一个就返回
		boolean b = arrayList.remove("高适");
		System.out.println("删除元素高适是否成功:" + b);//删除元素高适是否成功:true
  • void clear():删除集合中的所有元素,此集合仍旧存在,集合元素长度变0
		//清空元素
		arrayList.clear();
		System.out.println(arrayList.size());//0
 boolean removeAll(Collection<?> c)--- 差集

从集合中删除一个指定的集合元素:
删除A集合中存在B集合中的所有相同的元素,如果有删除返回True

ArrayList<String> list1 = new ArrayList<String>();
		list1.addAll(Arrays.asList("周杰伦", "郭德刚", "张学友", "于适"));

		ArrayList<String> list2 = new ArrayList<String>();
		list2.addAll(Arrays.asList("周杰伦", "王一博", "张学友", "吴磊"));

		// boolean removeAll(Collection<?> c)--- 差集
		boolean b = list1.removeAll(list2);
		System.out.println(b);//true
		System.out.println(list1);//[郭德刚, 于适]
		System.out.println(list2);//[周杰伦, 王一博, 张学友, 吴磊]
boolean retainAll(Collection<?> c)   --交集 

保留集合A和集合B中相同的元素,删除不同的元素,谁调用操作的是就是谁

ArrayList<String> list1 = new ArrayList<String>();
		list1.addAll(Arrays.asList("周杰伦", "郭德刚", "张学友", "于适"));

		ArrayList<String> list2 = new ArrayList<String>();
		list2.addAll(Arrays.asList("周杰伦", "王一博", "张学友", "吴磊"));
	
//		boolean retainAll(Collection<?> c)   --交集
		boolean result=list1.retainAll(list2);
		System.out.println(result);//true
		System.out.println(list1);//[周杰伦, 张学友]
		System.out.println(list2);//[周杰伦, 王一博, 张学友, 吴磊]

1.5 其他方法

clone()  

克隆一个集合,得到的一个长度,个数,内容,顺序完全一致的集合,单是赋值了一份

ArrayList<String> list1 = new ArrayList<String>();
		list1.addAll(Arrays.asList("z周杰伦", "g郭德刚", "d张学友", "y于适"));
		System.out.println("原集合:" + list1);//原集合:[z周杰伦, g郭德刚, d张学友, y于适]

		// list.clone() 克隆一个集合,得到的一个长度,个数,内容,顺序完全一致的集合,单是赋值了一份
		ArrayList<String> cloneList = (ArrayList<String>) list1.clone();
		System.out.println("克隆的集合:" + cloneList);//克隆的集合:[z周杰伦, g郭德刚, d张学友, y于适]
sort()

对list中的内容进行排序,需要自定义排序规则

// list.sort() 对list中的内容进行排序,需要自定义排序规则
		list1.sort(new Comparator<String>() {
			@Override
			public int compare(String o1, String o2) {
				return o1.compareTo(o2);
			}
		});

		System.out.println(list1);//[d张学友, g郭德刚, y于适, z周杰伦]

List和Array转换

  • object[] toArray()直接返回一个Object[]数组:
// object[] toArray()
		Object[] objects = list1.toArray();
		System.out.println("objects:"+Arrays.toString(objects));//objects:[d张学友, g郭德刚, y于适, z周杰伦]
  •  T[] toArray(T[] a)传入一个类型相同的Array,List内部自动把元素复制到传入的Array中:
// T[] toArray(T[] a)以较长的长度为
		String[] arrayList = list1.toArray(new String[0]);
		System.out.println("arrayList:"+Arrays.toString(arrayList));//arrayList:[d张学友, g郭德刚, y于适, z周杰伦]

2. 遍历

2.1 for循环

和数组类型,我们要遍历一个List,完全可以用for循环根据索引配合get(int)方法遍历:

        

ArrayList<String> arrayList = new ArrayList<String>();
		arrayList.addAll(Arrays.asList("李白", "高适", "李清照", "杜甫"));

		// 1.遍历
		for (int i = 0; i < arrayList.size(); i++) {
			System.out.print(arrayList.get(i) + " ");
		}
运行结果:李白 高适 李清照 杜甫 

2.2 增强for循环

        只要实现了Iterable接口的集合类都可以直接用for each循环来遍历,Java编译器本身并不知道如何遍历集合对象,但它会自动把for each循环变成Iterator的调用,原因就在于Iterable接口定义了一个Iterator<E> iterator()方法,强迫集合类必须返回一个Iterator实例。

ArrayList<String> arrayList = new ArrayList<String>();
		arrayList.addAll(Arrays.asList("李白", "高适", "李清照", "杜甫"));
		// 2.for each
		for (String str : arrayList) {
			System.out.print(str + "_");
		}
运行结果:李白_高适_李清照_杜甫_

2.3 迭代器

        是访问数据的模型,主要用来遍历Colllection集合。使用迭代器Iterator来访问List。Iterator本身也是一个对象,但它是由List的实例调用iterator()方法的时候创建的。Iterator对象知道如何遍历一个List,并且不同的List类型,返回的Iterator对象实现也是不同的,但总是具有最高的访问效率。

Iterator对象有两个方法:

  • boolean hasNext()判断是否有下一个元素,
  • E next()返回下一个元素。

因此,使用Iterator遍历List代码如下:

// 3.迭代器
		// 3.1 获取当前集合的“普通的”迭代器对象
		Iterator<String> it = arrayList.iterator();
		// 通过迭代器对象判断是否存在下一个元素
		while(it.hasNext()) {
			System.out.print(it.next()+" ");//获取下一个元素
		}
		System.out.println();

运行结果:李白 高适 李清照 杜甫 

使用List迭代器(ListIterator)遍历:

        继承了Iterator,是List集合派系所特有的迭代器,遍历方式与Iterator遍历类似,但也有其特殊之处:它内部含有hasPrevious()方法和previous()方法,可以配合使用进行倒序遍历,前提是必须要先正序遍历让指针移至最后,否则倒叙遍历没有效果

//3.2获取当前集合的"List迭代器"
		//从指定的下标开始正序遍历元素
		ListIterator<String> listIt = arrayList.listIterator(0);
		while(listIt.hasNext()) {
			System.out.print(listIt.next()+" ");//获取下一个元素
		}
		System.out.println();
		//逆序遍历集合
		ListIterator<String> listIt1 = arrayList.listIterator(4);
		while(listIt1.hasPrevious()) {
			System.out.print(listIt1.previous()+" ");
		}
运行结果:
李白 高适 李清照 杜甫 
杜甫 李清照 高适 李白 
Iterator的remove 
  • ①在使用迭代器的remove()操作时,会将更新后的modCount给expectedModCount,两者会得到同步,但是在调用集合的remove()方法后,两个不会进行同步,进而导致在checkForComodification()校验时不通过,抛出java.util.ConcurrentModificationException异常。
  • ②所以,在单线程下使用迭代器是没有问题的,但是在多线程下同时操作集合就不允许了,可以通过fail-fast快速失败机制,快速判断是否存在同时操作问题。因此,集合在多线程下使用是不安全的。
     Collection c2 = new ArrayList();
            c2.add("abc");
            c2.add("def");
            c2.add("xyz");
     
            Iterator it2 = c2.iterator();
            while(it2.hasNext()){
                Object o = it2.next();
                // 删除元素
                // 删除元素之后,集合的结构发生了变化,应该重新去获取迭代器
                // 但是,循环下一次的时候并没有重新获取迭代器,所以会出现异常:java.util.ConcurrentModificationException
                // 出异常根本原因是:集合中元素删除了,但是没有更新迭代器(迭代器不知道集合变化了)
                //c2.remove(o); // 直接通过集合去删除元素,没有通知迭代器。(导致迭代器的快照和原集合状态不同。)
                // 使用迭代器来删除可以吗?
                // 迭代器去删除时,会自动更新迭代器,并且更新集合(删除集合中的元素)。
                it2.remove(); // 删除的一定是迭代器指向的当前元素。
                System.out.println(o);
            }
     
            System.out.println(c2.size()); //0
    

3. 初始化

无参初始化

  • 无参构造初始化:内部数组初始化成空数组
//		1.无参构造初始化:内部数组初始化成空数组
		ArrayList<String> list1 =new ArrayList<String>();

有参初始化

  • ArrayList(int initialCapacity)初始化成长度为指定容量的Object[]数组
//		2.ArrayList(int initialCapacity)初始化成长度为指定容量的Object[]数组
		ArrayList<String> list2 =new ArrayList<String>(10);
		System.out.println(list2.size());//0
  • ArrayList(Collection<? extends E> c) 内部的数组被集合进行初始化
    //		3.ArrayList(Collection<? extends E> c) 内部的数组被集合进行初始化
    		ArrayList<String> list3=new ArrayList<String>(Arrays.asList("abc","def"));
    		System.out.println(list3);//[abc, def]

五、ArrayList底层源码解析及扩容机制(⭐)

List的主要实现类,底层使用Object[]存储,适用于频繁的查找工作,线程不安全。

【特点】 

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

【数据结构】

底层是基于数组实现的,随着元素的增加而动态扩容

使用场景:适合数据连续性遍历,读多写少的场景


		ArrayList<String> list = new ArrayList<String>(0);
		
		//第一次往无参构造的数组list中添加元素的时候,数组首次扩容10
		list.add("zkt");
		list.add("zkt");
		list.add("zkt");
		list.add("zkt");
		list.add("zkt");
		list.add("zkt");
		list.add("zkt");
		list.add("zkt");
		list.add("zkt");
		list.add("zkt");
		
	
		//...
		//后面每次容量不足时扩容为容量的1.5倍
		list.add("张kt");
		System.out.println(list);//[zkt, zkt, zkt, zkt, zkt, zkt, zkt, zkt, zkt, zkt, 张kt]
		
		
		list.addAll(Arrays.asList("张三","李四","王五"));

扩容机制:

  1. 当需要添加元素时,ArrayList会先判断当前容量是否足够,如果不足够,则需要进行扩容操作。
  2. 扩容操作的第一步是调用方法,该方法用于确定扩容后的最小容量。
  3. 如果ArrayList为空,则选择默认容量和传入容量的较大值作为新容量。
  4. 然后调用ensureExplicitCapacity方法,该方法用于判断是否需要进行扩容操作
  5. ensureExplicitCapacity方法中,修改操作次数加1,然后判断如果需要的最小容量大于当前容量,则进行扩容操作。
  6. 扩容操作的第二步是调用grow方法,该方法用于计算新的容量并进行扩容。
  7. 首先获取当前的容量,然后计算新的容量,一般是原容量的1.5倍,并向上取整。
  8. 如果新容量小于所需的最小容量,则直接使用最小容量作为新容量。
  9. 如果新容量超过了最大容量限制,则使用最大容量作为新容量。
  10. 最后,通过调用Arrays.copyOf方法,将原始数组拷贝到新数组中,实现扩容。

底层代码:

属性:

private static final int DEFAULT_CAPACITY = 10;// 默认的初始容量是10
// 空元素数组
private static final Object[] EMPTY_ELEMENTDATA = {};//有参构造所创建
// 默认容量的空数组
private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {};//无参构造所创建的
// 存储元素的数组 
transient Object[] elementData;//底层为Object类型的数组,存储的元素都在此。
// 集合容量
private int size;

构造方法 :

空参构造:

构造一个初始容量的空数组。

// 1.构造一个初始容量的空数组。
public ArrayList() {
    this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA;
}

单参(int)构造:

如果参数initialCapacity大于0,给elementData 初始化为长度为initialCapacity的数组;若等于0,初始化为一个长度为0的数组;若小于0则抛出异常

// 2.构造具有指定初始容量的数组。
public ArrayList(int initialCapacity) {
    if (initialCapacity > 0) {
        this.elementData = new Object[initialCapacity];
    } else if (initialCapacity == 0) {
        this.elementData = EMPTY_ELEMENTDATA;
    } else {
        throw new IllegalArgumentException("Illegal Capacity: "+ initialCapacity);
    }
}

单参(Collection)构造:

传入容器参数,将c容器内的内容赋值给elementData,如果c无内容,则将一个空数组赋值给elementData,同时初始化数组长度size

//3.构造一个包含指定集合的元素的列表,按照它们由集合的迭代器返回的顺序
public ArrayList(Collection<? extends E> c) {
    // 将集合构造中的集合对象转成数组
    Object[] a = c.toArray();
    if ((size = a.length) != 0) {
        if (c.getClass() == ArrayList.class) {
            // 类型为ArrayList则直接赋值
            elementData = a;
        } else {
            //如果不一样,使用Arrays的copyOf方法进行元素的拷贝
            elementData = Arrays.copyOf(a, size, Object[].class);
        }
    } else {
        // replace with empty array.
        elementData = EMPTY_ELEMENTDATA;
    }
}

扩容源码:

进入add方法:

先执行ensureCapacityInternal确保数组elementData的容量足够,再将e添加进elementData

// 添加一个元素
public boolean add(E e) {
        ensureCapacityInternal(size + 1);  // Increments modCount!!
        elementData[size++] = e;//e 操作对象;  elementData 底层操作的数组;size 默认大小0
        return true; 
}

进入ensureCapacityInternal方法:

其中calculateCapacity方法返回elementData需要的最低长度,该整数作为参数传入ensureExplicitCapacity方法中,用于判断是否需要扩容

// 1.主体函数
private void ensureCapacityInternal(int minCapacity) {
    // 对容量进行判断
    ensureExplicitCapacity(calculateCapacity(elementData, minCapacity));
}

进入calculateCapacity方法:

如果elementData 为空,则返回max(10, minCapacity),否则直接返回minCapacity

// 2.通过最小容量和默认容量求出较大值 (主要用于第一次扩容:如果是无参构造方式创建的数组对象,第一次添加元素的时候扩容到大小为10)
private static int calculateCapacity(Object[] elementData, int minCapacity) {
    if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
        return Math.max(DEFAULT_CAPACITY, minCapacity);
    }
    return minCapacity;
}

进入ensureExplicitCapacity方法:

如果elementData所需的最小长度大于目前实际的长度,则进入grow方法,进行扩容

// 3.判断是否需要进行扩容
private void ensureExplicitCapacity(int minCapacity) {
    // 实际修改集合次数+1 (在扩容的过程中没用,主要是用于迭代器中)
    modCount++;
    // 判断当前最小容量是否大于数组长度
    if (minCapacity - elementData.length > 0)
        // 将计算出来的容量传递给核心扩容方法
        grow(minCapacity);
}

进入grow方法:

新数组长度=max(旧数组长度*1.5,数组所需最低长度)。如果新数组长度过大(大于Integer.MAX_VALUE - 8),进入hugeCapacity

// 4.核心扩容方法
private void grow(int minCapacity) {
    // 记录数组的实际长度
    int oldCapacity = elementData.length;
    // 核心扩容算法,原容量的1.5倍
    int newCapacity = oldCapacity + (oldCapacity >> 1);
    // 判断新容量是否小于当前最小容量(第一次调用add方法必然小于)
    if (newCapacity - minCapacity < 0)
        newCapacity = minCapacity;
    
    // 判断新容量是否大于最大数组长度,如果
    if (newCapacity - MAX_ARRAY_SIZE > 0)
        
        // 条件满足就计算出一个超大容量
        newCapacity = hugeCapacity(minCapacity);

    // 调用数组工具类方法,创建一个新数组,将新数组的地址赋值给elementData
    elementData = Arrays.copyOf(elementData, newCapacity);
}

进入hugeCapacity方法:

如果minCapacity溢出,抛出异常,否则返回一个合法的大的值

private static int hugeCapacity(int minCapacity) {
        if (minCapacity < 0) // overflow
            throw new OutOfMemoryError();
        return (minCapacity > MAX_ARRAY_SIZE) ?
            Integer.MAX_VALUE :
            MAX_ARRAY_SIZE;
    }

 总结:

ArrayList扩容的本质就是计算出新的扩容数组的size后实例化,并将原有数组内容复制到新数组中去。(不是原数组,而是新数组然后给予数组对象地址)。

默认情况下,新的容量会是原容量的1.5倍。 新容量=旧容量右移一位(相当于除于2)在加上旧容量

  ArrayList 的底层是用动态数组来实现的我们初始化一个ArrayList 集合还没有添加元素时,其实它是个空数组,只有当我们添加第一个元素时,内部会调用扩容方法并返回最小容量10,也就是说ArrayList 初始化容量为10。 当前数组长度小于最小容量的长度时(前期容量是10,当添加第11个元素时就就扩容),便开始可以扩容了,ArrayList 扩容的真正计算是在一个grow()里面,新数组大小是旧数组的1.5倍,如果扩容后的新数组大小还是小于最小容量,那新数组的大小就是最小容量的大小,后面会调用一个Arrays.copyof方法,这个方法是真正实现扩容的步骤。

小结

Collection(I) 单例,一个一个的存储
       List集合(I)有序,可重复
               实现类:ArrayList:
                            所属的包: java.util.*
                            构造方法:

  •                                     ArrayList<E>()
  •                                     ArrayList<E>(int init)
  •                                     ArrayList<E>(Collection<E> c)

                              添加元素:

  •                                     boolean add(E)---添加元素到集合的末尾
  •                                     void  add(index,E)---添加元素到指定的位置
  •                                     boolean addAll(Collection c)---添加集合到集合的末尾
  •                                     boolean addAll(int index,Collection c)---添加集合到集合的末尾

                                 查看元素:  

  •                                     boolean contains(E) --- 是否包含某个元素
  •                                     int indexOf(Object c)--- 查找某个元素存在的下标位置
  •                                     E get(int index)---查找对应的下标元素
  •                                     boolean isEmpty()
  •                                     size() -- 查看元素个数
  •                                     List<E> subList(int for,int end)--截取集合
  •                                     boolean equals()--

                              修改元素:oldValue set(int index,E newElement)
                              删除元素:  

  •                                         E remove(int index)
  •                                         boolean remove(Object obj)
  •                                         clear()
  •                                         boolean removeAll(Collection c)
  •                                         boolean retainAll(Colleciont c)

                              其他方法:

  •                                   clone()克隆一个集合
  •                                   sort(Comparator) 排序
  •                                   Object[] toArray()
  •                                   T[] toArray(T[])

                                  
                              元素遍历:

  •                               for
  •                               foreach
  •                               普通迭代器 Iterator it = list.iterator();
  •                               hasNext
  •                               next
  •                               list迭代器:ListIterator it = list.listIterator(9);

                     
                      LinkedList
                      Vector
                      Stack
       Set集合(I)无序,不可重复
       Queue队列(I)先进先出

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

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

相关文章

UniApp 开发微信小程序教程(二):下载安装微信开发者工具

文章目录 一、微信开发者工具简介二、下载安装微信开发者工具1. 下载微信开发者工具步骤&#xff1a; 2. 安装微信开发者工具Windows 系统&#xff1a;Mac 系统&#xff1a; 3. 配置微信开发者工具登录微信开发者工具&#xff1a;新建项目&#xff1a; 4. 预览和调试预览&#…

电商还存在错位竞争空间吗?

“上链接试了&#xff0c;十几分钟&#xff0c;成本5块的东西卖1块5了。”今年618前期&#xff0c;某个电商平台上线了自动跟价功能&#xff0c;有一个卖家尝试了一会儿之后赶紧关了。 又一个618&#xff0c;平台、商家、消费者们又迎来了一次狂欢。只是与往年不同的是&#x…

从2-3-4树开始理解红黑二叉树(JAVA代码手撸版)

经典的红黑二叉树在新增/删除数据时维持自平衡始终对应着一个2-3-4 树。本文只关注2-3-4 对应的经典红黑二叉树。 暂时不考虑 2-3 树对应的左倾红黑二叉树。 背景知识 2-3-4 树简介 一棵 2-3-4 树的结点分为 内部结点 (internal nodes) 和 叶子结点 (leaf nodes) &#xff0c;…

KVB交易平台:国内三大交易所(上海、深圳、北京)的概要与分析

概述&#xff1a; 上海证券交易所&#xff08;上交所&#xff09;、深圳证券交易所&#xff08;深交所&#xff09;和北京证券交易所&#xff08;北交所&#xff09;是中国大陆三大主要证券交易所。以下是对这三个交易所的比较分析&#xff1a; 一、基本概况 1. 上海证券交易所…

零基础入门AI:四步快速搭建本地的编程助手

由于种种原因&#xff0c;小伙伴们在写代码时&#xff0c;不一定能用上Github Copilot&#xff1b;或者由于代码安全的原因&#xff0c;不能使用外网的编程助手。今天我就介绍一种利用开源大语言模型在本地搭建编程助手的方案&#xff1a; IDE插件&#xff1a;Continue本地开源…

DB9母头接口定义485

在通信技术中&#xff0c;DB9接口广泛应用于串行通信&#xff0c;尤其是在RS232和RS485标准中。虽然DB9接口最常见于RS232通信&#xff0c;但通过适当的引脚映射&#xff0c;它也可以用于RS485通信。本文将详细介绍如何定义和使用DB9母头接口进行RS485连接。 DB9母头接口简介 …

探索FlowUs息流:个人和团队知识管理解决方案|FlowUs稳定保障你的笔记安全无忧

FlowUs息流&#xff1a;稳定运营保障你的笔记安全无忧 在知识管理工具的选择上&#xff0c;稳定性是用户最关心的问题之一。FlowUs息流以其稳定的运营记录&#xff0c;为用户提供了一个可靠的工作环境。我们深知&#xff0c;一个知识管理平台的稳定性直接影响到团队的生产力和…

echarts引入百度地图vue3(大屏项目中缩放点偏移到左上角,解决代码在最后)

实际开发中的问题&#xff0c;遇到了大屏做了自适应&#xff0c;为非标准文档流之后&#xff0c;在缩放时不是以鼠标当前位置缩放的&#xff0c;而是偏移到左上角。 向百度地图提了工单也没解决&#xff0c;同一套适应方案用cesium地图时缩放没问题&#xff1a; 先看看效果&am…

动态规划——达拉崩吧

1、题目链接 174. 地下城游戏 2、题目分析 假如说我们正向推状态转移方程&#xff0c;很难推出来&#xff0c;因为这道题有“加血”的说法&#xff0c;只能依靠后面的值判断前面所需要的血量&#xff0c;也就是说&#xff0c;如果正向的dp表示从起点出发&#xff0c;到达&…

CSS新手入门笔记【导入方法、选择器介绍、选择器优先级、属性详细介绍、盒子模型】

目录 一、目的与优势二、CSS导入方式三、语法结构四、选择器类型基本选择器组合选择器伪类与伪元素属性选择器 六、选择器优先级总结 六、CSS属性1. 字体与文本属性2. 背景属性3. 尺寸与盒模型属性4. 布局与定位5. 列表样式6. 边框与轮廓7. 文本装饰与效果8. 动画与过渡9. 伪类…

多模块存储器

随着计算机技术的发展&#xff0c;处理的信息量越来越多&#xff0c;对存储器的速度和容量要求也越来越高&#xff1b;而且随着CPU性能的不断提高、IO设备数量不断增加&#xff0c;导致主存的存取速度已经称为了整个计算机系统的性能瓶颈。这就要求我们必须提高主存的访问速度。…

Linux基础命令大全(详解版)

Linux基础命令&#xff08;详解版&#xff09; 文章目录 Linux基础命令&#xff08;详解版&#xff09;1.Linux的目录结构**2.Linux路径的描述方式**3.Linux命令基础格式4.ls命令 隐藏文件、文件夹5.pwd命令6.cd命令 特殊路径符7.mkdir命令 文件操作命令8.touch命令9.cat命令10…

package.json简介

1、package.json简介 通过 npm init 初始化一个项目&#xff0c;会生成3个目录/文件&#xff0c; node_modules, package.json和 package.lock.json。其中package-lock.json文件是为了锁版本。 2、package.json常用属性 1&#xff09;name name是项目的名称&#xff0c;命名…

远程桌面无法连接,远程桌面连接登录没有成功

在信息技术领域&#xff0c;远程桌面连接作为一种便捷的管理和操作工具&#xff0c;对于企业和个人用户而言都具有极其重要的价值。然而&#xff0c;在实际使用过程中&#xff0c;远程桌面无法连接的问题时常出现&#xff0c;这不仅影响了工作效率&#xff0c;还可能对数据安全…

javascript--类型检测 type of 和 instanceof

类型判断 1、typeof2、instanceof**instanceof 的原理** 3、constructor 1、typeof typeof在检测null、object、array、data的结果中都是object&#xff0c;所以无法用来区分这几个类型的区别。 <script>let a ["123",123,false,true,Symbol(1),new Date(),n…

小兔鲜02

elementplus自动按需引入 elementplus主题色定制 安装sass npm install sass -D要替换的主题色内容&#xff1a; /* 只需要重写你需要的即可 */ forward element-plus/theme-chalk/src/common/var.scss with ($colors: (primary: (// 主色base: #27ba9b,),success: (// 成功…

lock-锁的概念

锁的简介 锁是计算机协调多个进程或线程并发访问某一资源的机制&#xff08;避免发生资源争抢&#xff09; 在并发环境下&#xff0c;多个线程会对同一个资源进行争抢&#xff0c;可能会导致数据不一致的问题。为了解决这一问题&#xff0c;需要通过一种抽象的锁来对资源进行…

第 402 场 LeetCode 周赛题解

A 构成整天的下标对数目 I 计数&#xff1a;遍历 h o u r s hours hours &#xff0c;记录 h o u r s [ i ] % 24 hours[i]\%24 hours[i]%24 的出现次数 class Solution {public:long long countCompleteDayPairs(vector<int>& hours) {vector<int> cnt(24);…

【内含优惠码】重磅发售!《2023年度中国量化投资白皮书》(纸质版)

这是可以公开了解量化行业图景的&#xff0c;为数不多资料。 简介 《2023年度中国量化投资白皮书》由宽邦科技、华泰证券、金融阶、华锐技术、AMD、阿里云、英迈中国等多家机构联合发起编写&#xff0c;并于2024年6月15日正式发布&#xff0c;全书公17万字6大章节勾勒最新量化…

【Unity | Editor强化工具】项目备忘录工具

经常会被美术和策划同事反复询问某几个问题&#xff0c;每次都要翻Wiki链接给他们&#xff0c;非常折磨人&#xff0c;所以做了个可以在Unity内部显示备忘录的小工具&#xff0c;能够减少一些查找成本&#xff08;另外我觉得&#xff0c;让他们养成查看Unity内触手可及的信息的…