集合类存储在任何编程语言中都是很重要的内容,只因有这样的存储数据结构才让我们可以在内存中轻易的操作数据,那么在Java中这些存储类集合结构都有哪些?内部实现是怎么样?有什么用途呢?下面分享一些我的总结

集合类存储结构的种类及其继承关系图

图中只列出了比较关键的继承关系,在Java中所有的集合类都实现Collection接口,在直接的继承关系中主要分为两大接口:一个是列表实现的List接口,另一个是集合实现的Set接口。在列表中最为常用的实现类是ArrayList和LinkedList。在集合中最为常用的实现类则是HashSet和LinkedHashSet。虽然这些具体的实现有所不同,但所包含的操作却大致相同。Collection又扩展了Iterator接口为各个实现类提供遍历功能。下面我们分别描述各个实现类实现原理和用途。

注: 只能是引用类型,要想存储基本的数据类型需要使用对应的引用类型结构

列表和集合的区别

实现了List接口的列表与于实现了Set接口的集合之间对比如下:

  1. 列表中允许存储重复元素而集合则不允许存储元素。
  2. 元素加入列表中的顺序是固定的而集合则是无序的,所以集合在遍历的时候并不是按照添加顺序输出的。
  3. 列表中的元素可以通过索引进行访问而集合不能。

ArrayList

ArrayList可以说是在Java开发的过程中是常用的存储结构了,通过名字大致可以猜到它的内部实现其实是通过数组来存储的。那究竟是不是这么回事呢?我们来一探究竟。

public class ArrayList<E> extends AbstractList<E>
implements List<E>, RandomAccess, Cloneable, java.io.Serializable
{
private static final long serialVersionUID = 8683452581122892189L; private static final int DEFAULT_CAPACITY = 10; private static final Object[] EMPTY_ELEMENTDATA = {}; private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {}; transient Object[] elementData; // non-private to simplify nested class access private int size; public ArrayList(int initialCapacity) {
if (initialCapacity > 0) {
this.elementData = new Object[initialCapacity];
} else if (initialCapacity == 0) {
this.elementData = EMPTY_ELEMENTDATA;
} else {
throw new IllegalArgumentException("Illegal Capacity: "+
initialCapacity);
}
} public ArrayList() {
this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA;
}

查看源码我们发现这样一行代码transient Object[] elementData; 并且在对ArrayList进行初始化的时候也对这个属性进行了赋值操作。内部果然是采用Object数组存储的。既然内部是数组实现的,操作也和数组差不多,为什么不直接用数组呢?在Java中数组一旦定义长度既不可更改,而在ArrayList中数组的元素是可以随意添加的,在ArrayList内部默认使用的数组长度为10,当对List添加的元素个数超过10之后,会对数组进行扩容和对数据复制。每次在添加元素的时候,如果数组满了,就会触发扩容操作计算出一个新的数组容量并使用Arrays.copyOf操作(内部是通过System.arraycopy来操作的)对数据进行整体的复制

ArrayList既然内部是使用数组来实现的,也就继承了数组的特性:支持快速查找,但是对于添加和删除操作来说数组的性能会慢一些,在需要频繁进行添加和删除元素的场景下,会引起频繁的数组扩容和数据移动,降低性能。所以在读多写少的场景下非常合适。

LinkedList

和ArrayList同属于List的一种实现方式,区别于ArrayList,但是内部的实现却和ArrayList从名字上能猜测出来一样,是使用链表来实现内部存储的。下面来看下源码

public class LinkedList<E>
extends AbstractSequentialList<E>
implements List<E>, Deque<E>, Cloneable, java.io.Serializable
{
transient Node<E> first; transient Node<E> last; public E getLast() {
final Node<E> l = last;
if (l == null)
throw new NoSuchElementException();
return l.item;
} public E removeFirst() {
final Node<E> f = first;
if (f == null)
throw new NoSuchElementException();
return unlinkFirst(f);
} public void addFirst(E e) {
linkFirst(e);
} private static class Node<E> {
E item;
Node<E> next;
Node<E> prev; Node(Node<E> prev, E element, Node<E> next) {
this.item = element;
this.next = next;
this.prev = prev;
}
}

查看源码我们发现在LinkedList的内部维护了一个内部实现类Node结构用于存储列表中的元素,查看Node的代码不难看出,Node实现了是一个双向链表。既然是链表,那么LinkedList也就继承了链表的特性:查询性能低,但支持快速的添加和删除操作。故在需要进行频繁添加和删除操作的场景下,更为适用。与ArrayList是互斥的。

对列表中的元素进行判重操作

有时候我们需要判断列表中是否包含一个元素,我们会调用相应类型的Contains方法,而在Contains的实现内部则是使用存储的数据类型的equals方法来进行判等操作的。我们以ArrayList的源码为例

public boolean contains(Object o) {
return indexOf(o) >= 0;
}
public int indexOf(Object o) {
return indexOfRange(o, 0, size);
} int indexOfRange(Object o, int start, int end) {
Object[] es = elementData;
if (o == null) {
for (int i = start; i < end; i++) {
if (es[i] == null) {
return i;
}
}
} else {
for (int i = start; i < end; i++) {
if (o.equals(es[i])) {
return i;
}
}
}
return -1;
}

当被查找的元素不为null时,会调用元素的equals方法进行判等操作。在存储自定义类型的时候,比如自定义类Person,在判断元素是否存在的时候会调用Person的equals方法,默认情况下会比较两个元素的地址,对于不同的Person类实例,地址也不相同,这是没有意义的。所以我们需要进行重写equals方法来实现对Person的判等操作。

HashMap

在我们开始讲集合的实现类之前,先来看一下HashMap这个结构,在集合实现类中无论是HashSet和LinkedHashSet内部的实现方式均是依赖于HashMap的。

//HashSet的内部实现部分代码
public class HashSet<E>
extends AbstractSet<E>
implements Set<E>, Cloneable, java.io.Serializable
{
private transient HashMap<E,Object> map; private static final Object PRESENT = new Object(); public HashSet() {
map = new HashMap<>();
} public HashSet(int initialCapacity, float loadFactor) {
map = new HashMap<>(initialCapacity, loadFactor);
} //LinkedHashSet内部实现部分代码
public class LinkedHashSet<E>
extends HashSet<E>
implements Set<E>, Cloneable, java.io.Serializable { public LinkedHashSet(int initialCapacity, float loadFactor) {
super(initialCapacity, loadFactor, true);
}

HashMap实现Map<K,V>接口,存储的是键值对的映射关系,并不属于Collection接口的实现类,在HashMap的内部使用的Hash表来存储数据的,具体Hash表怎么一回事,还是通过源码来研究研究吧

//HashMap的主要源码
public class HashMap<K,V> extends AbstractMap<K,V>
implements Map<K,V>, Cloneable, Serializable { static final int DEFAULT_INITIAL_CAPACITY = 1 << 4; static final float DEFAULT_LOAD_FACTOR = 0.75f; static class Node<K,V> implements Map.Entry<K,V> {
final int hash;
final K key;
V value;
Node<K,V> next; Node(int hash, K key, V value, Node<K,V> next) {
this.hash = hash;
this.key = key;
this.value = value;
this.next = next;
}
} transient Node<K,V>[] table; static final int hash(Object key) {
int h;
return (key == null) ? 0 : (h = key.hashCode()) ^ (h >>> 16);
} final float loadFactor;
public HashMap(int initialCapacity, float loadFactor) {
if (initialCapacity < 0)
throw new IllegalArgumentException("Illegal initial capacity: " +
initialCapacity);
if (initialCapacity > MAXIMUM_CAPACITY)
initialCapacity = MAXIMUM_CAPACITY;
if (loadFactor <= 0 || Float.isNaN(loadFactor))
throw new IllegalArgumentException("Illegal load factor: " +
loadFactor);
this.loadFactor = loadFactor;
this.threshold = tableSizeFor(initialCapacity);
}
public V put(K key, V value) {
return putVal(hash(key), key, value, false, true);
}

源码中我们可以看到在HashMap的内部存储着一个内部类实例数组Node<K,V>[],默认情况下,这个数组的长度为16。除此之外还有一个Hash函数和一个装填因子。他们是做什么用的呢?我们先着重看一下Put方法,每次我们向HashMap的实例中添加元素的时候,都会对Key使用Hash函数计算出来一个整数hash值,然后和数组的长度进行运算得出一个索引值,这个索引值就是该元素应该在数组中的位置。如果这个位置并没有元素存在,则直接放置在该位置,如果该位置的元素已经存在,也被称为哈希碰撞,则使用单向链表的方式将元素连接起来。内部类Node<K,V>就是一个单向链表。当数组的容量超过(填装因子*容量)的时候,意味着hash表的存储非常臃肿,哈希碰撞会增多,会降低程序的性能(这里hash函数计算出hash值并且运算得到位置时间复杂度为O(1),如果在相同位置出现碰撞的次数越多就需要在链表中进行查找元素了,链表查找元素的时间复杂度是O(N),这会大大降低程序的性能),这个时候就需要对数组进行扩容,对所有元素进行迁移,这个过程也叫reHash。

我们在初始化HashMap的时候可以指定容量和填装因子,容量一定要是2的幂,填装因子的默认值为0.75。但是这里我不建议初始化的时候主动去设置这些值。因为这些值设置的是否合理直接影响到程序的性能,容量设置的大,浪费空间,容量设置的小,会导致哈希碰撞的次数增多,而且一旦超过了阈值(容量*填装因子)还会导致扩容和数据迁移,这对程序的性能会大打折扣。

HashMap给我们遍历它存储的元素暴露出一些有用的方法,最为常用的则为:entrySet() 方法返回键值对作为值的集合;keySet() 方法返回键的集合;并且在HashMap中的Key和Value都允许为null。

HashSet

上面描述了List及其实现类的实现方式和用途,接下来我们对比看一下集合及其实现类的原理及用途。在某些场景下,我们存储的元素中不需要有重复,这个时候集合就派上了用场,例如维护爬虫爬取的链接。

我们先来看下集合的第一个主要实现类HashSet。

前面讲HashMap原理的时候,我们说过HashSet的内部存储就是靠HashMap来实现的,HashMap<K,V>是键值对的形式,而集合实现类并不存在这样的关系,所以在使用HashMap的过程中对于集合类而言,Value是不存储值的,默认情况下是个Object类型的null值。

private static final Object PRESENT = new Object();

public boolean add(E e) {
return map.put(e, PRESENT)==null;
}

HashSet在使用方式上除了和列表对比的那几点不同之外没有任何区别,具体的用途也可以根据它的特点来选择合适的使用场景。

LinkedHashSet

LinkedHashSet继承自HashSet,只不过LinkedHashSet可以保证存入的顺序和取出的顺序是一样,是一个有序的集合。它是如何在HashSet的基础上实现的呢?老规矩,源码走起

Node<K,V> newNode(int hash, K key, V value, Node<K,V> e) {
LinkedHashMap.Entry<K,V> p =
new LinkedHashMap.Entry<>(hash, key, value, e);
linkNodeLast(p);
return p;
}
transient LinkedHashMap.Entry<K,V> head; transient LinkedHashMap.Entry<K,V> tail; static class Entry<K,V> extends HashMap.Node<K,V> {
Entry<K,V> before, after;
Entry(int hash, K key, V value, Node<K,V> next) {
super(hash, key, value, next);
}
}
private void linkNodeLast(LinkedHashMap.Entry<K,V> p) {
LinkedHashMap.Entry<K,V> last = tail;
tail = p;
if (last == null)
head = p;
else {
p.before = last;
last.after = p;
}
}

原来是这样:LinkedHashSet重写了newNode方法并且在内部维护了一个新的Entry类和一个双向链表,在每次创建新节点的时候都会对head,tail指针进行更新,就是这个双向链表保证集合元素在遍历的时候输出的结果就是插入时的顺序。除了这一点之外,用法和HashSet并无不同。

对集合中的元素进行判重操作

当我们需要判断一个元素是否存在于集合中或者是向集合中添加重复元素时,除了需要像列表一样重写equals方法外,还需要重写hashCode方法。在HashMap的内部,首先比对HashCode,如果这个值相等才会去比较equals。默认情况下HashCode是对地址的编码,和equals一样都和地址有关系,不重写的话这种比较是没有意义的。

final Node<K,V> getNode(int hash, Object key) {
Node<K,V>[] tab; Node<K,V> first, e; int n; K k;
if ((tab = table) != null && (n = tab.length) > 0 &&
(first = tab[(n - 1) & hash]) != null) {
if (first.hash == hash && // always check first node
((k = first.key) == key || (key != null && key.equals(k))))
return first;
if ((e = first.next) != null) {
if (first instanceof TreeNode)
return ((TreeNode<K,V>)first).getTreeNode(hash, key);
do {
if (e.hash == hash &&
((k = e.key) == key || (key != null && key.equals(k))))
return e;
} while ((e = e.next) != null);
}
}
return null;
}

总结

本节介绍了关于Java泛型集合存储类一些常见的实现类及其原理和应用,上面介绍的所有的实现类均是线程不安全的。所以在多线程模式下访问需要注意这一点,并且需要对其操作进行额外的防护。

Java中关于泛型集合类存储的总结的更多相关文章

  1. java中的泛型2--注意的一些问题和面试题

    前言 这里总结一下泛型中需要注意的一些地方和面试题,通过面试题可以让你掌握的更清楚一些. 泛型相关问题 1.泛型类型引用传递问题 在Java中,像下面形式的引用传递是不允许的: ArrayList&l ...

  2. 夯实Java基础系列13:深入理解Java中的泛型

    目录 泛型概述 一个栗子 特性 泛型的使用方式 泛型类 泛型接口 泛型通配符 泛型方法 泛型方法的基本用法 类中的泛型方法 泛型方法与可变参数 静态方法与泛型 泛型方法总结 泛型上下边界 泛型常见面试 ...

  3. [JavaCore]JAVA中的泛型

    JAVA中的泛型 [更新总结] 泛型就是定义在类里面的一个类型,这个类型在编写类的时候是不确定的,而在初始化对象时,必须确定该类型:这个类型可以在一个在里定义多个:在一旦使用某种类型,在类方法中,那么 ...

  4. Java 中的泛型详解-Java编程思想

    Java中的泛型参考了C++的模板,Java的界限是Java泛型的局限. 2.简单泛型 促成泛型出现最引人注目的一个原因就是为了创造容器类. 首先看一个只能持有单个对象的类,这个类可以明确指定其持有的 ...

  5. java中的泛型(转)

    什么是泛型? 泛型(Generic type 或者 generics)是对 Java 语言的类型系统的一种扩展,以支持创建可以按类型进行参数化的类.可以把类型参数看作是使用参数化类型时指定的类型的一个 ...

  6. 【Java入门提高篇】Day14 Java中的泛型初探

    泛型是一个很有意思也很重要的概念,本篇将简单介绍Java中的泛型特性,主要从以下角度讲解: 1.什么是泛型. 2.如何使用泛型. 3.泛型的好处. 1.什么是泛型? 泛型,字面意思便是参数化类型,平时 ...

  7. Java中的泛型 --- Java 编程思想

    前言 ​ 我一直都认为泛型是程序语言设计中一个非常基础,重要的概念,Java 中的泛型到底是怎么样的,为什么会有泛型,泛型怎么发展出来的.通透理解泛型是学好基础里面中非常重要的.于是,我对<Ja ...

  8. Java中的泛型 - 细节篇

    前言 大家好啊,我是汤圆,今天给大家带来的是<Java中的泛型 - 细节篇>,希望对大家有帮助,谢谢 细心的观众朋友们可能发现了,现在的标题不再是入门篇,而是各种详细篇,细节篇: 是因为之 ...

  9. Java中的泛型 (上) - 基本概念和原理

    本节我们主要来介绍泛型的基本概念和原理 后续章节我们会介绍各种容器类,容器类可以说是日常程序开发中天天用到的,没有容器类,难以想象能开发什么真正有用的程序.而容器类是基于泛型的,不理解泛型,我们就难以 ...

随机推荐

  1. [Spring cloud 一步步实现广告系统] 22. 广告系统回顾总结

    到目前为止,我们整个初级广告检索系统就初步开发完成了,我们来整体回顾一下我们的广告系统. 整个广告系统编码结构如下: mscx-ad 父模块 主要是为了方便我们项目的统一管理 mscx-ad-db 这 ...

  2. OSI七层网络模型与TCP/IP四层模型

    1.OSI七层结构图: 2.TCP/IP四层结构图: 3.各层对应的协议 4.OSI七层和TCP/IP四层的区别 OSI网络模型和TCP/IP网络模型对应关系: 5.交换机工作在OSI的哪一层 如果有 ...

  3. hive动态分区与静态分区

    测试目的:1.分区表的动态分区与静态分区2.每层数据,数据流向,数据是否在每层都保留一份测试结果:1.动态分区/静态分区略2.每层表的数据都会保留,因此在生产上odm层的数据是可以删除的(不管是内表还 ...

  4. 03 requests模块基础

    1. requests 模块简介 什么是requests 模块 requests模块是python中原生的基于网络请求的模块,功能强大,用法简洁高效.在爬虫领域中占据着半壁江山的地位.requests ...

  5. F#周报2019年第34期

    新闻 高效的F#,提示与技巧 Fable 社区资源 Visual Studio提示与技巧:为.NET增加生产力 无风险地尝试Compositional IT的培训包--如果没有增加任何价值,可以得到完 ...

  6. think in java 泛型

    曾几何时,我们对java的泛型充满了好奇,但是感觉用起来有很爽,但又会在spring类型泛型的地方,遇到问题. 我第一次的遇到泛型是在使用别人的BaseDao的时候,这是一个java封装hiberna ...

  7. java 获取真实ip和根据ip获取ip所在地区

    import com.alibaba.fastjson.JSON; import javax.servlet.http.HttpServletRequest; import java.io.ByteA ...

  8. C++ switch注意事项(陷阱)

    话不多说,直接上代码 int a; printf("请输入一个整数:"); scanf("%d", &a); switch (a) { : printf ...

  9. Nginx总结(四)基于域名的虚拟主机配置

    前面讲了如何安装配置Nginx,大家可以去这里看看nginx系列文章:https://www.cnblogs.com/zhangweizhong/category/1529997.html 今天要说的 ...

  10. 到底什么是故事点(Story Point)?

    故事点是一个度量单位,用于表示完成一个产品待办项或者其他任何某项工作所需的所有工作量的估算结果. 当采用故事点估算时,我们为每个待办项分配一个点数.待办项估算结果的原生数据并不重要,我们只关注最后得到 ...