java源码----集合系列1----ArrayList,linkedList

Arraylist

基础信息

底层是一个object数组

Arraylist 是java里面Collection  标准的一个集合,其底层是一个object数组。当new一个空参的ArrayList的时候,会默认生成一个空数组。

Arraylist上限是 Integer.MAX_VALUE - 8(Integer.MAX_VALUE  =  2^31-1);

超过上限会报内存溢出

这里为什么是Integer.MAX_VALUE-8  ,源码上的解释就一句话 

Some VMs reserve some header words in an array

 一些虚拟机会在数组中保留一些头字。解释起来就是:

Java对象在堆内存中的存储布局可以分为三部分:对象头(object header),实例数据(Instance Data)和对齐填充(Padding);

对象头包括

Mark Word:用于对象自身的运行时数据存储,如HashCode,GC分代年龄,锁状态标志,线程持有的锁,偏向线程ID和偏向时间戳等;
Klass Pointer:对象指向它类元数据的指针,JVM通过这个指针长度来确定对象是哪个类的实例。
数组长度(只有数组对象才有):记录数组对象的长度。

8 bytes(Mark Word的最大占用) + 8 bytes(Klass Pointer的最大占用) + 4 bytes(数组长度)+ 8 bytes(引用指针的最大占用:数组中存放的是对象的引用) + 4 bytes(padding:为了方便寻址,JVM要求对象大小要求是8的倍数,不够就填充) ==32 bytes

  而int类型  1位占4个byte, 8 int(整数) 就等于32 bytes(字节),所以是Integer.MAX_VALUE-8。但在实际使用时因为考虑到虚拟机堆大小,实际arraylist大小要远远小于Integer.MAX_VALUE-8 。

ArrayList  默认长度是10,使用Arrays.copyOf()进行扩容,每次=扩容size是 原size的一半  (oldCapacity + (oldCapacity >> 1));

 Arrays.copyOf()源码其实是调用了System.arraycopy 

 System.arraycopy 是原生方法。

public static native void arraycopy(Object src,  int  srcPos,
                                    Object dest, int destPos,
                                    int length);

相当于新建了一个空数组,再用原生方法把老数组的值复制进去。从这一点可以看出ArrayList每次扩容都需要创建新的数组,并且将原数组的值复制进新的数组,新增的速度相对要慢

查询的时候先校验是否存在数组下标越界

然后从数组中获取

LinkeList

LinkeList底层是个链表

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值