[置顶] JAVA部分集合实现方式的简单概述

博客说明:以下是集合实现的简单概述,如果想更加具体的了解某类集合的实现,移步----->JAVA源码阅读之AbEf;
集合命名规则: AbEf:  Ab代表存储结构,Ef代表逻辑结构
必知的一个native方法:
  • java.lang.System类中
  • public static native void arraycopy(Object src,  int  srcPos, Object dest, int destPos, int length);
  • 功能:将src中的srcPos~srcPos+length-1的数据复制到dest中的destPos~destPos+length-1位置处;

顺序存储结构实现的集合(即数组实现)

A、B、C、D是按照底层实现的相似性进行划分的

PartA

ArrayList:
  • 内部存储了一个Object[];提供随机访问;
  • ArrayList中数据的移动大多是利用System.arraycopy();方法实现,该方法是native方法;
  • 扩容;
    • 默认大小是10;
    • 最大容量Integer.MAX_VALUE - 8;
    • 扩容触发条件:当前要求容量大小大于当前容量大小
    • 正常扩容newCapacity = oldCapacity + (oldCapacity >> 1);
    • 使用System.arraycopy将数据全部移动到新的数组中
  • 缩容:手动调用trimToSize()方法,会缩到当前数组实际存储的数据大小
Vector:
  • Vector与ArrayList共同点
    • 两者实现的功能相同,方法的实现也基本一致;
    • 都具有SubList内部类和Itr迭代器内部类
    • 都能够扩容和缩容
  • Vector与ArrayList区别:
    • 线程安全:Vector对大部分的方法使用了关键字Synchronized声明
    • 额外的%element%方法:这些方法会被Stack类所使用;
Stack:
  • Stack主要使用的是Vector<E>的方法进行具体的实现,自身没有状态量
  • 线程安全:继承Vector,故也是线程安全的
  • pop方法:使用elementAt+removeElementAt
  • push方法:使用addElement
  • search方法:从后往前搜索调用lastIndexOf

PartB

ArrayDeque:
  • 存储有一个Obejct[]数组;数组长度必须是2的幂次,因为这样就方便取余运算;
  • 有一个head和tail指针;
  • tail =(tail+1)&(elements.length-1)
  • head =(head+1)&(elements.length-1)
  • size = (tail-head)&(elements.length-1)
  • 扩容:
    • 正常扩容后大小为原数组长度2倍
    • 利用System.arraycopy方法实现
PriorityQueue:
  • 类中存储有一个Object[]数组
  • size为当前数组中存储的数据的大小
  • 存入的数据类型要求实现了Comparable接口
  • 数据的添加和删除操作伴随着调堆的操作;
  • 具有扩容功能:
    • 扩容大小:当前大小的2倍或者1.5倍
    • 扩容条件:准备存入的数据量大于当前Obejct数组的长度;

PartC

HashMap:
  • 内部存储了一个Node<K,V>[] table;
  • 内部有定义一个Node内部类和TreeNode内部类;后者是一颗红黑树
  • 数组的下标(散列值)等于hash(key)&n-1;
  • 会自动扩容:
    • 默认大小是16;大小必须为2的幂
    • 最大容量是1<<30
    • 填装因子为0.75
    • 扩容触发条件:一旦当前存储数据大于门限(容量*填装因子)
    • 正常扩容newCapacity = oldCapacity<<1;
    • 每个桶中的数据分成两拨一波继续留在原散列值下面,一波移动到原散列值+oldCapacity位置处
  • 不能自动减容;
  • 红黑树的利用:
    • 当一个桶中的元素大于某个固定值,则将该桶中的数据由链表存储结构,转换成红黑树存储结构;
    • 在此之后的所有操作交由红黑树进行处理;
    • 在扩容的时候,对红黑树进行拆分的过程中,如果最后元素个数小于固定值,则将红黑树存储结构转换为普通链表存储结构
HashTable:
  • 基本与HashMap类似;
  • 线程安全:HashTable的大部分方法使用了Synchronized关键字进行标记
  • 没有红黑树:HashTable对桶中数据的存储没有使用红黑树,因此效率相对于hashmap会差一点
  • 不支持存储null:会抛出异常
HashSet:
  • 内部存储了一个HashMap<E,Object>变量
  • 一个简单版本的HashMap;
  • 只使用了key而没有使用value部分,value永远存的都是固定的Object

PartD

String
  • 内部存储的是一个final char[]字符数组;
  • 因此不能修改它指向的字符数组
StringBuidler
  • 内部存储的是一个char[]字符数组
  • 因此可以修改里面的字符
  • 对于数组的拷贝大多依赖于System.arraycopy的方法
  • 扩容
    • 默认大小为16
    • 容量最大值:Integer.MAX_VALUE
    • 正常情况扩容大小:int newCapacity = value.length * 2 + 2;
    • 扩容触发条件:准备存储的字符数大于当前数组的长度
  • 缩容:
    • 为当前存储数据的大小
StringBuffer
  • 与Stringbuilder类似
  • 线程安全:方法大多使用了Synchronized关键字标记

链式存储结构实现的集合(即链表实现)

A、B是按照底层实现的相似性进行划分的

PartA

TreeMap:
  • 定义了一个内部类Entry<K,V>;该节点定义了这课红黑树的存储结构;
  • 存储有该树的根节点;要删除这颗树只需要将root指向null即可;GC之后便会回收该红黑树,造成其它节点的引用不可达
  • 每次插入、删除一个树中节点时;都需要对这颗红黑树进行一定的调整;
TreeSet:
  • 内部存储了一个TreeSet<E,Object>对象
  • 一个简单版本的TreeMap;
  • 只使用了key而没有使用value部分,value永远存的都是固定的Object

PartB

LinkedList:
  • 内部存储一个双向链表;所有的操作都是对该链表进行操作 


















你可能感兴趣的:(java,集合,源代码,概述,底层实现)