- 1.常用的集合类有哪些?
- 2.List,Set,Map三者的区别?
- 3.常用集合框架底层数据结构
- 4.哪些集合类是线程安全的?
- 5.迭代器Iterator是什么
- 6.Java集合的快速失败机制 “fail-fast”和安全失败机制“fail-safe”是什么?
- 7.如何边遍历边移除 Collection 中的元素?
- 8.Array 和 ArrayList 有什么区别?
- 9.comparable 和 comparator的区别?
- 10.Collection和Collections有什么区别?
- 11.遍历一个List有哪些不同的方式?
- 12.ArrayList的扩容机制
- 13.ArrayList和LinkedList的区别是什么?
- 14.ArrayList和Vector的区别是什么?
- 15.简述ArrayList、Vector、LinkedList 的存储性能和特性?
- 16.说一下HashSet的实现原理
- 17.HashSet如何检查重复?(HashSet是如何保证数据不可重复的?)
- 18.HashSet与HashMap的区别
- 19.HashMap在JDK1.7和JDK1.8中有哪些不同?HashMap的底层实现?
- 20.HashMap的长度为什么是2的幂次方
- 21.HashMap的put方法的具体流程?
- 22.HashMap的扩容操作是怎么实现的?
- 23.HashMap默认加载因子为什么选择0.75?
- 24.为什么要将HashMap链表中转红黑树的阈值设为8?为什么不一开始直接使用红黑树?
- 25.HashMap是怎么解决哈希冲突的?
- 26.HashMap为什么不直接使用hashCode()处理后的哈希值直接作为table的下标?
- 27.关于HashMap的key一些问题
- 28.HashMap多线程导致死循环问题
- 29.ConcurrentHashMap底层具体实现知道吗?
- 30.HashTable的底层实现知道吗?
- 31.HashMap、ConcurrentHashMap及Hashtable的区别
- 32.Java集合的常用方法
32.Java集合的常用方法
- 初始值为16,负载因子为0.75,阈值为负载因子*容量
resize()
方法是在hashmap
中的键值对大于阀值时或者初始化时,就调用resize()
方法进行扩容。- 每次扩容,容量都是之前的两倍
- 扩容时有个判断
e.hash & oldCap
是否为零,也就是相当于hash值对数组长度的取余操作,若等于0,则位置不变,若等于1,位置变为原位置加旧容量。
源码如下:
final Node<K,V>[] resize() {
Node<K,V>[] oldTab = table;
int oldCap = (oldTab == null) ? 0 : oldTab.length;
int oldThr = threshold;
int newCap, newThr = 0;
if (oldCap > 0) {
if (oldCap >= MAXIMUM_CAPACITY) { //如果旧容量已经超过最大值,阈值为整数最大值
threshold = Integer.MAX_VALUE;
return oldTab;
}else if ((newCap = oldCap << 1) < MAXIMUM_CAPACITY &&
oldCap >= DEFAULT_INITIAL_CAPACITY)
newThr = oldThr << 1; //没有超过最大值就变为原来的2倍
}
else if (oldThr > 0)
newCap = oldThr;
else {
newCap = DEFAULT_INITIAL_CAPACITY;
newThr = (int)(DEFAULT_LOAD_FACTOR * DEFAULT_INITIAL_CAPACITY);
}
if (newThr == 0) {
float ft = (float)newCap * loadFactor;
newThr = (newCap < MAXIMUM_CAPACITY && ft < (float)MAXIMUM_CAPACITY ?
(int)ft : Integer.MAX_VALUE);
}
threshold = newThr;
@SuppressWarnings({"rawtypes","unchecked"})
Node<K,V>[] newTab = (Node<K,V>[])new Node[newCap];
table = newTab;
if (oldTab != null) {
for (int j = 0; j < oldCap; ++j) {
Node<K,V> e;
if ((e = oldTab[j]) != null) {
oldTab[j] = null;
if (e.next == null)
newTab[e.hash & (newCap - 1)] = e;
else if (e instanceof TreeNode)
((TreeNode<K,V>)e).split(this, newTab, j, oldCap);
else {
Node<K,V> loHead = null, loTail = null;//loHead,loTail 代表扩容后在原位置
Node<K,V> hiHead = null, hiTail = null;//hiHead,hiTail 代表扩容后在原位置+旧容量
Node<K,V> next;
do {
next = e.next;
if ((e.hash & oldCap) == 0) { //判断是否为零,为零赋值到loHead,不为零赋值到hiHead
if (loTail == null)
loHead = e;
else
loTail.next = e;
loTail = e;
}
else {
if (hiTail == null)
hiHead = e;
else
hiTail.next = e;
hiTail = e;
}
} while ((e = next) != null);
if (loTail != null) {
loTail.next = null;
newTab[j] = loHead; //loHead放在原位置
}
if (hiTail != null) {
hiTail.next = null;
newTab[j + oldCap] = hiHead; //hiHead放在原位置+旧容量
}
}
}
}
}
return newTab;
}
本站链接:https://www.mianshi.online,如需勘误或投稿,请联系微信:lurenzhang888
点击面试手册,获取本站面试手册PDF完整版