ArrayList、Vector、HashMap、Hash…

ArrayList、Vector、HashMap、HashSet的默认初始容量、加载因子、扩容增量

原文转自http://www.cnblogs.com/xiezie/p/5511840.html

这里要讨论这些常用的默认初始容量和扩容的原因是:

当底层实现涉及到扩容时,容器或重新分配一段更大的连续内存(如果是离散分配则不需要重新分配,离散分配都是插入新元素时动态分配内存),要将容器原来的数据全部复制到新的内存上,这无疑使效率大大降低。

加载因子的系数小于等于1,意指  即当 元素个数 超过 容量长度*加载因子的系数 时,进行扩容。

另外,扩容也是有默认的倍数的,不同的容器扩容情况不同。

List 元素是有序的、可重复

ArrayList、Vector默认初始容量为10

Vector:线程安全,但速度慢

    底层数据结构是数组结构

    加载因子为1:即当 元素个数 超过 容量长度 时,进行扩容

    扩容增量:原容量的 1倍

      如 Vector的容量为10,一次扩容后是容量为20

ArrayList:线程不安全,查询速度快

    底层数据结构是数组结构

    扩容增量:原容量的 0.5倍 1

      如 ArrayList的容量为10,一次扩容后是容量为16

Set(集) 元素无序的、不可重复。

HashSet:线程不安全,存取速度快

     底层实现是一个HashMap(保存数据),实现Set接口

     默认初始容量为16(为何是16,见下方对HashMap的描述)

     加载因子为0.75:即当 元素个数 超过 容量长度的0.75倍 时,进行扩容

     扩容增量:原容量的 1 倍

      如 HashSet的容量为16,一次扩容后是容量为32

Map是一个双列集合

HashMap:默认初始容量为16

     (为何是16:16是2^4,可以提高查询效率,另外,32=16<<1       -->至于详细的原因可另行分析,或分析源代码)

     加载因子为0.75:即当 元素个数 超过 容量长度的0.75倍 时,进行扩容

     扩容增量:原容量的 1 倍

      如 HashSet的容量为16,一次扩容后是容量为32

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: ArrayListHashMap都有扩容的机制。 ArrayList的扩容是在添加元素时,如果当前元素个数已经达到了底层数组的容量,就会创建一个新的数组,将原数组中的元素复制到新数组中,并将新元素添加到新数组中。新数组的容量一般是原数组容量的1.5倍。 HashMap的扩容是在添加元素时,如果当前元素个数已经达到了负载因子(默认为.75)乘以当前数组容量的大小,就会创建一个新的数组,将原数组中的元素重新计算哈希值后,放入新数组中。新数组的容量一般是原数组容量的两倍。 需要注意的是,扩容会涉及到数组的复制和元素的重新计算哈希值,所以会有一定的性能开销。因此,在预知数据量较大的情况下,可以通过设置初始容量来减少扩容的次数,提高性能。 ### 回答2: ArrayListHashMap都是常用的数据结构,在实际开发中使用频率极高,而它们之所以被广泛使用,与它们内部的扩容机制密不可分。 ArrayList是数组实现的线性表,底层是通过数组来存储数据的,而在实际使用中,数组的长度是不确定的,因此需要进行扩容。ArrayList的扩容机制比较简单,当数组已满,需要添加新元素时,会申请一个新的长度为原来1.5倍的数组,并将原数组中的元素复制到新数组中。在大多数情况下,这种扩容机制是能够满足需求的,因为ArrayList中的操作大都是在末尾进行的。 而HashMap是基于哈希表实现的,其内部是由一个数组+链表/红黑树的结构来实现的。其扩容机制是,当HashMap中的元素数量超过阈值,即负载因子的设定值,会创建一个新的容量是原来的两倍的数组,并将原数组中的元素重新计算哈希值,然后重新插入到新数组中。需要注意的是,这个重新计算哈希值和重新插入元素的过程是比较耗时的,因此需要充分考虑扩容阈值的设置。 总之,ArrayListHashMap在扩容时,都要考虑合理设置扩容因子以及重新分配数组后元素的复制和重新插入等问题,以保证它们的性能和稳定性。 ### 回答3: ArrayListHashMap是Java中非常常用的两个集合类。它们在使用过程中都要考虑到其容量的问题,因为当空间不足时,它们需要进行扩容操作来添加更多的元素。下面我们将分别来介绍ArrayListHashMap的扩容规则。 ArrayList的扩容规则: ArrayList是一个数组实现的集合类,其内部维护了一个Object数组作为底层容器。当在ArrayList中添加元素时,如果当前容量不足,那么ArrayList会按照一定的扩容规则来扩容数组的容量。 默认情况下,ArrayList的初始容量为10个,每次扩容时,原来的容量会自动增加50%。例如,如果当前ArrayList的容量是8,需要添加第9个元素时,ArrayList会自动将当前容量扩展为12。而当需要添加的元素个数超过了当前容量,那么ArrayList会直接将当前容量扩展到至少需要的大小。例如,如果当前ArrayList的容量是10,需要添加第15个元素时,ArrayList会直接将容量扩展为15。在进行ArrayList扩容时,Java虚拟机会新建一个长度为新容量的Object数组,然后将原数组中的元素复制到新数组中,最后将新数组作为底层容器。 HashMap的扩容规则: HashMap是一个基于哈希表的映射集合类,其内部维护了一个Entry[]数组作为底层容器。当在HashMap中添加元素时,如果当前容量不足,那么HashMap会按照一定的扩容规则来扩容数组的容量。 默认情况下,HashMap的初始容量为16个,扩容因子为0.75。也就是说,当HashMap中元素的个数达到容量的75%时,就会进行扩容操作。在扩容时,HashMap会新建一个长度为原数组两倍的Entry[]数组,然后将原数组中的元素重新放入新数组中。需要注意的是,由于哈希函数针对哈希表长度进行取模运算,不与哈希表长度互质的直接值也很少,所以哈希表的容量必须扩展为2的幂次方。如果容量为非2的幂次方,则在取模运算时需要进行额外的计算,会降低哈希表的访问效率。 总体上来说,ArrayListHashMap的扩容规则都是在容量不足时,新建一个更大的底层数组,并将原数组中的原有元素复制到新数组中,再将新数组作为底层容器。对于ArrayList而言,每次扩容都是原来容量的50%,而对于HashMap而言,容量 超过0.75倍时就进行扩容,且扩容后长度必须是2的幂次方。扩容过程虽然会增加运算时间,但是带来的好处是能够使数据结构在快速插入元素的同时,也能保证空间的充分利用和高效的查询和访问速度。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值