集合总结

集合



常用集合关系图


各种集合的区别

集合分为单列集合和双列集合两种:

一.单列集合:

Collection的结构图

Collection是单列集合的顶级接口:

其中有三类集合:

1.List(ArrayList,LinkedList,Vector等)

有序的可以重复的集合,JDK1.6和JDK1.7的时候创建集合时初始容量是10,而JDK1.8中默认容量是0,首次添加元素不超过10时,容量变为10。

List的加载因子系数<=1,即当元素个数超过(容器长度*加载因子系数)时,进行扩容

①ArrayList集合是动态数组的数据结构实现的,它的随机访问和遍历的效率比较高,在需要频繁的读取集合中的元素的时候,推荐使用ArrayList,但是增删操作要影响数组内的其他数据的下标,所以增删操作的效率比较慢。

ArrayList每次扩增容量为原本的1.5倍,JDK1.8之后算法为oldCapacity+(oldCapacity >>1)。

ArrayList的扩容上限约21亿,int的最大值。

②LinkedList集合是双向的链表的数据结构实现,没有下标,通过链来连接数据,在非首尾的增加和删除操作时,LinkedList比ArrayList的效率要高,推荐使用LinkedList,但是查询的时候,每次都要从首位移动指针往后依次查找,所以查询的效率比较慢。

③Vector集合与ArrayList集合类似,内部都是维护一个数组,只不过前者在方法上加上了synchronized关键字、线程安全、效率低,后者线程不安全,但是效率高。

Vector每次扩增容量为原来的两倍。

2.Set(HashSet等)

除了TreeSet集合外元素无序,不允许元素重复。

Set的扩容量为原来的2倍,加载因子为0.75。

①HashSet集合是基于HashMap实现的,HashSet底层使用HashMap来保存元素,HashMap是数组和链表的数据结构(下面HashMap具体介绍),HashSet和HashMap的初始容量都是16

3.Queue/Deque

①Queue队列,队列是一种特殊的线性表,它只允许在表的前端进行删除操作,而在表的后端进行插入操作,LinkedList类实现了Queue接口,可以吧LinkedList当成Queue来用

②Deque双端队列,可以在两端进行插入和删除操作

二.双列集合:

Map的结构示意图

1.Map(HashMap,Hashtable等)

①HashMap集合线程非安全,使用键值对(key-value)的方式存储数据,允许key和value为null,键(key)不能重复,值(value)可以重复,它和HashSet是基于哈希表的链表散列的数据结构,即底层是数组和链表(模拟指针)实现的,到了JDK1.8后变成了数组+链表+红黑树。

HashMap底层是通过一个transient(防止反序列化) Node[]table node数组实现的,接下来单个Node数据类型:是HanshMap静态内部类。静态内部类中有一个成员变量:

Node<K,V>next;

 通过该成员变量,其实底层用的是单向链表,性能低


HashMap的结构示意图


HashMap的执行流程


HashMap 基于 Hash 算法实现的,我们通过 put(key,value)存储,get(key)来获取。当传入 key 时,HashMap 会根据 key. hashCode() 计算出 hash 值,根据 hash 值将 value 保存在 bucket 里。当计算出的 hash 值相同时并且equals比较的值不同时,我们称之为 hash 冲突,HashMap 的做法是用链表和红黑树存储相同 hash 值的 value。当 hash 冲突的个数比较少时,使用链表否则使用红黑树。

在向HashMap中添加数据的时候,会先调用hashCode方法计算并比较hash值,当hash相同的情况在调用equals方法比较,当两个均不相同时判定集合中不存在,然后将数据插入到集合中。两个对象的hash值相同,但是不一定是同一个对象,也就是说自身的值可能会不相同。

(例如:String a ="通话";String b="重地";这两个字符串生成的hash值同为1179395,但是其自身的值却不相同)


HashMap和Entry的关系

从图中可以看出: 

1)HashMap继承于AbstractMap类,实现了Map接口。Map是"key-value键值对"接口,AbstractMap实现了"键值对"的通用函数接口。 

2)HashMap是通过"拉链法"实现的哈希表。它包括几个重要的成员变量:table, size, threshold, loadFactor, modCount。

table是一个Entry[]数组类型,而Entry实际上就是一个单向链表。哈希表的"key-value键值对"都是存储在Entry数组中的。

size是HashMap的大小,它是HashMap保存的键值对的数量。 

threshold是HashMap的阈值,用于判断是否需要调整HashMap的容量。threshold的值="容量*加载因子",当HashMap中存储数据的数量达到threshold时,就需要将HashMap的容量加倍。

loadFactor就是加载因子。 

modCount是用来实现fail-fast机制的。

LinkedHashMap:底层也是一个Entry<k,v>数组,接下来单个Entry数据类型

Entry<K,V>before,after;

②Hashtable线程安全,是线程安全的,但是其同步锁是全局锁,效率很低,所以Hashtable现在是保留类不建议使用。

单线程的情况下HashMap效率高,推荐使用;多线程的情况下可以使用java.util.concurrent包下的ConcurrentHashMap替代,它之前采用Segment方式的分段同步锁,将所有的数据分割成几部分的数据区域,一个区域一把锁,访问不同区域时不会收到影响,JDK1.8版本之后采用优化后的synchronized,将每一个数据分别锁住,不同数据间的访问不会收到影响,效率大大增加,并且线程是安全的,多线程并发情况下很推荐使用!

双列集合的五种遍历方式

1)通过map.keySey();获取所有的key的Set集合,然后可以通过增强for自动迭代,也可以获取迭代器iterator然后用while循环进行手动遍历

2)通过map.values();获得所有的value的集合,返回值为Collecton,然后用增强for自动迭代

3)(推荐)通过map.entrySet();获取所有的键值对的Set集合,Set集合中保存的是每一个键值对(Map.Entry),然后可以通过增强for自动迭代,也可以获取迭代器iterator然后用while循环进行手动遍历

三.关于迭代器

迭代器是Iterator接口:该接口中有三个核心方法 ,维护指针可以向下移动(next),移动到指定位置后,取出当前位置的元素(next),以及重置指针操作(remove)。

为什么数组和集合可以使用for循环进行迭代遍历?

解析:所有的数组和集合都实现了Iterable接口,Iterable接口中只有一个方法,iterator方法返回值类型是Iterator类型,我们将思路转到Iterator,我们发现该接口三个方法。hasNext,next和remove,最主要的是hasNext和next。他们在底层帮我们去维护的可以被迭代数组或者集合的迭代策略。

四.JDK版本差异更新

排序算法

线程安全的集合

集合扩容的算法

五.集合在特定场景下的应用方案

最近浏览可以选取LinkedList

先进先出可以考虑Queue队列

先进后出可以考虑Stack,递归,压栈 ,弹栈 

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,684评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,143评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,214评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,788评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,796评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,665评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,027评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,679评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,346评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,664评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,766评论 1 331
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,412评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,015评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,974评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,073评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,501评论 2 343

推荐阅读更多精彩内容

  • Java集合类主要有2大分支,Collection及Map。Collection体系如下: Map体系如下: **...
    Huang远阅读 379评论 0 0
  • 剖析面试最常见问题之Java集合 1.说说list、set和map的区别?    1.list是有序可重复;   ...
    血武行者阅读 326评论 0 0
  • Java集合类可用于存储数量不等的对象,并可以实现常用的数据结构如栈,队列等,Java集合还可以用于保存具有映射关...
    小徐andorid阅读 1,915评论 0 13
  • 经过十一篇文章的分析,终于把一些主要的集合类的实现原理分析完了。本文,我们将对之前分析的知识点做一次总结。 集合框...
    swz_android阅读 932评论 0 1
  • 躺在向日葵花海上的飞毯 随风摇曳 空气中弥漫着阳光和花海的气息 那味道被蒸发出来了 仿佛都能被看见、被摸到 天空的...
    三石与三心阅读 118评论 0 0