1.Java里面elementData 和elementDats有什么区别吗?
2.面试题:ArrayList扩容时扩容多少?
3.arraylist为ä»ä¹çº¿ç¨ä¸å®å
¨
4.List LinkedList HashSet HashMap底层原理剖析
Java里面elementData 和elementDats有什么区别吗?
直接看源码,源码可以看到的解析是,ArrayList有三个构造函数:
无参构造:
无参构造时,源码Obeject数组elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA;
当其为有参构造时:会有两种情况:
给定初始容量构造:
源码具体逻辑如下:
当 传入的解析初始容量initialCapacity > 0为真时,创建一个大小为initialCapacity的源码空数组,并将引用赋给elementData;
当 传入的解析短信宝接入源码初始容量initialCapacity = 0为真时,将空数组EMPTY_ELEMENTDATA赋给elementData;
当 传入的源码初始容量initialCapacity < 0为真时,直接抛出IllegalArgumentException异常。解析
面试题:ArrayList扩容时扩容多少?
大家好,源码我是解析你们的小米!今天要和大家一起来探讨一个在Java面试中经常被问到的源码问题:“ArrayList扩容时扩容多少?”相信很多小伙伴都在面试中遇到过这个问题,那么接下来,解析我就为大家详细解析一下这个问题,源码梦幻新源码希望能够帮助大家在面试中游刃有余!解析
了解ArrrayList的源码内部实现
在深入解析扩容策略之前,我们首先要了解一下ArrayList的内部实现原理。ArrayList是Java集合框架中的一个动态数组,它可以根据需要动态地增加或减少元素。ArrayList的底层是通过数组实现的,当数组容量不足以存放新增的元素时,就需要进行扩容操作。
扩容策略简介
ArrayList在扩容时,并不是每次新增一个元素就扩容一次,这样效率会很低。相反,源码多卖它采取了一种“倍增”策略,即当数组容量不够用时,它会将当前容量翻倍。这样做的好处是,在一次扩容操作中,可以一次性扩充一大块内存,减少了频繁扩容带来的性能损耗。
源码分析
ArrayList的扩容逻辑实际上是由ensureCapacityInternal方法来完成的。我们一起来看一下这段源码:
从上面的代码中,我们可以看到,在grow方法中,新的全网营销源码容量(newCapacity)是通过将旧容量(oldCapacity)右移一位(即除以2),然后再加上旧容量得到的。这样就实现了容量的翻倍扩容策略。
理解扩容的触发条件
在源码分析的基础上,我们来总结一下ArrayList扩容的触发条件:
需要注意的是,虽然数组会根据倍增策略进行扩容,但也并不是无限制地扩容下去的。在源码中,有一个MAX_ARRAY_SIZE的限制,如果计算得到的新容量超过了这个值,就会进行特殊处理。
END
通过对ArrayList扩容策略的源码分析和解释,我们可以得出ArrayList在扩容时采用了倍增策略,书本翻页源码每次扩容都会将当前容量翻倍,从而有效地减少了频繁扩容带来的性能损耗。同时,也要注意到MAX_ARRAY_SIZE的限制,防止无限制地扩容。掌握了这些知识,相信在面试中回答关于ArrayList扩容策略的问题时,大家已经游刃有余了!
arraylist为ä»ä¹çº¿ç¨ä¸å®å ¨
é¦å 说ä¸ä¸ä»ä¹æ¯çº¿ç¨ä¸å®å ¨ï¼çº¿ç¨å®å ¨å°±æ¯å¤çº¿ç¨è®¿é®æ¶ï¼éç¨äºå éæºå¶ï¼å½ä¸ä¸ªçº¿ç¨è®¿é®è¯¥ç±»çæ个æ°æ®æ¶ï¼è¿è¡ä¿æ¤ï¼å ¶ä»çº¿ç¨ä¸è½è¿è¡è®¿é®ç´å°è¯¥çº¿ç¨è¯»åå®ï¼å ¶ä»çº¿ç¨æå¯ä½¿ç¨ãä¸ä¼åºç°æ°æ®ä¸ä¸è´æè æ°æ®æ±¡æã线ç¨ä¸å®å ¨å°±æ¯ä¸æä¾æ°æ®è®¿é®ä¿æ¤ï¼æå¯è½åºç°å¤ä¸ªçº¿ç¨å åæ´æ¹æ°æ®é ææå¾å°çæ°æ®æ¯èæ°æ®ã å¦å¾ï¼Listæ¥å£ä¸é¢æ两个å®ç°ï¼ä¸ä¸ªæ¯ArrayListï¼å¦å¤ä¸ä¸ªæ¯vectorã ä»æºç çè§åº¦æ¥çï¼å 为Vectorçæ¹æ³åå äºï¼synchronized å ³é®åï¼ä¹å°±æ¯åæ¥çææï¼sunå ¬å¸å¸æVectoræ¯çº¿ç¨å®å ¨çï¼èå¸æarraylistæ¯é«æçï¼ç¼ºç¹å°±æ¯å¦å¤çä¼ç¹ã 说ä¸åçï¼ç¾åº¦çï¼å¾å¥½ç解ï¼ï¼ ä¸ä¸ª ArrayList ï¼å¨æ·»å ä¸ä¸ªå ç´ çæ¶åï¼å®å¯è½ä¼æ两æ¥æ¥å®æï¼
1. å¨ Items[Size] çä½ç½®åæ¾æ¤å ç´ ï¼
2. å¢å¤§ Size çå¼ã
å¨å线ç¨è¿è¡çæ åµä¸ï¼å¦æ Size = 0ï¼æ·»å ä¸ä¸ªå ç´ åï¼æ¤å ç´ å¨ä½ç½® 0ï¼èä¸ Size=1ï¼
èå¦ææ¯å¨å¤çº¿ç¨æ åµä¸ï¼æ¯å¦æ两个线ç¨ï¼çº¿ç¨ A å å°å ç´ åæ¾å¨ä½ç½® 0ãä½æ¯æ¤æ¶ CPU è°åº¦çº¿ç¨Aæåï¼çº¿ç¨ B å¾å°è¿è¡çæºä¼ã线ç¨Bä¹åæ¤ ArrayList æ·»å å ç´ ï¼å 为æ¤æ¶ Size ä»ç¶çäº 0 ï¼æ³¨æå¦ï¼æ们å设çæ¯æ·»å ä¸ä¸ªå ç´ æ¯è¦ä¸¤ä¸ªæ¥éª¤å¦ï¼è线ç¨Aä» ä» å®æäºæ¥éª¤1ï¼ï¼æ以线ç¨Bä¹å°å ç´ åæ¾å¨ä½ç½®0ãç¶å线ç¨Aå线ç¨Bé½ç»§ç»è¿è¡ï¼é½å¢å Size çå¼ã
é£å¥½ï¼ç°å¨æ们æ¥çç ArrayList çæ åµï¼å ç´ å®é ä¸åªæä¸ä¸ªï¼åæ¾å¨ä½ç½® 0ï¼è Size å´çäº 2ãè¿å°±æ¯â线ç¨ä¸å®å ¨âäºã
示ä¾ç¨åºï¼
package test;
import java.util.ArrayList;
import java.util.List;
public class ArrayListInThread implements Runnable {
List<String> list1 = new ArrayList<String>();// not thread safe
// List<String> list1 = Collections.synchronizedList(new ArrayList<String>());// thread safe
public void run() {
try {
Thread.sleep((int)(Math.random() * 2));
}
catch (InterruptedException e) {
e.printStackTrace();
}
list1.add(Thread.currentThread().getName());
}
public static void main(String[] args) throws InterruptedException {
ThreadGroup group = new ThreadGroup("mygroup");
ArrayListInThread t = new ArrayListInThread();
for (int i = 0; i < ; i++) {
Thread th = new Thread(group, t, String.valueOf(i));
th.start();
}
while (group.activeCount() > 0) {
Thread.sleep();
}
System.out.println();
System.out.println(t.list1.size()); // it should be if thread safe collection is used.
}
}
List LinkedList HashSet HashMap底层原理剖析
ArrayList底层数据结构采用数组。数组在Java中连续存储,因此查询速度快,时间复杂度为O(1),插入数据时可能会慢,特别是需要移动位置时,时间复杂度为O(N),但末尾插入时时间复杂度为O(1)。数组需要固定长度,ArrayList默认长度为,最大长度为Integer.MAX_VALUE。在添加元素时,如果数组长度不足,则会进行扩容。JDK采用复制扩容法,通过增加数组容量来提升性能。若数组较大且知道所需存储数据量,可设置数组长度,或者指定最小长度。例如,设置最小长度时,扩容长度变为原有容量的1.5倍,从增加到。
LinkedList底层采用双向列表结构。链表存储为物理独立存储,因此插入操作的时间复杂度为O(1),且无需扩容,也不涉及位置挪移。然而,查询操作的时间复杂度为O(N)。LinkedList的add和remove方法中,add默认添加到列表末尾,无需移动元素,相对更高效。而remove方法默认移除第一个元素,移除指定元素时则需要遍历查找,但与ArrayList相比,无需执行位置挪移。
HashSet底层基于HashMap。HashMap在Java 1.7版本之前采用数组和链表结构,自1.8版本起,则采用数组、链表与红黑树的组合结构。在Java 1.7之前,链表使用头插法,但在高并发环境下可能会导致链表死循环。从Java 1.8开始,链表采用尾插法。在创建HashSet时,通常会设置一个默认的负载因子(默认值为0.),当数组的使用率达到总长度的%时,会进行数组扩容。HashMap的put方法和get方法的源码流程及详细逻辑可能较为复杂,涉及哈希算法、负载因子、扩容机制等核心概念。