Java集合简单解析
一、 Collection
1、 List
a、 ArrayList
b、 Vector
c、 LinkedList
首先要对List的三种实现进行一个简单的异同比较:
同:
*ArrayList和Vector都可以看做是一个可变的数组;
*ArrayList和LinkedList都属于线程不安全的类型;
*但是ArrayList和LinkedList都可以通过调用Colletions类里的SynchronizedList()来进行ArrayList和LinkedList的调用来保证线程的安全性;
异:
*ArrayList和Vector虽然都可以看做是一个可变的数组,但是Vector类中大部分函数都包含了Synchronized关键字保证了线程的安全性;
*ArrayList和LinkedList虽然都属于线程不安全的类型,但是LinkedList实际上是一种双向链表的存储集合,而ArrayList实际上是数组;
*ArrayList和Vector因为都是可变的数组,所以在数据查找上的效率是跟数组差不多,因为数组存储过程中会有明确的下表同时保存起来所以相对速度会更快一些,但是在数据插入的时候,在数组中插入数据实际上是牵扯到了内存的调动所以在数据插入时是相对较慢的,而Vector因为大部分方法都考虑了线程安全的问题,所以效率相对ArrayList更加缓慢一些;
*LinkedList因为是双向链表方式,所以实际上是通过<Node>节点来进行数据的存储的,并在存储一个节点的同时,记录它的前一个及后一个节点,并且在集合中标识出了首节点first和终节点last。所以在数据添加的时候实际上只要在对应节点添加指向并记录前后节点即可,在效率上远远高于ArrayList,但是在查询是,当LinkedList需要查询某一个下标的节点数据时,需要遍历整个集合,计算相对应的下标才可以查询出来,所以LinkedList的查询相对更加缓慢一些。
ArrayList的add方法表明是数组添加:
Vector的函数实现,表明是线程安全的:
LinkedList源码展示,表现是双向链表:
代码展示ArrayList和LinkedList插入和查询效率比较:
输出结果为:
这个结果有没有很惊讶呢?
2、 Set
a、 HashSet
b、 TreeSet
简单分析一下HashSet和TreeSet的异同
同:
*都是基于Map来进行数据的存储的
*都是线程不同步的
*都不能存储重复数据(因为都是通过Map的key值进行数据存储的,所以不允许存在重复数据)
*都只能通过Iterator来进行遍历
异:
*HashSet是一个无序的集合,TreeSet是一个有序的集合,并且可根据自然排序或者TreeSet创建时的Comparator进行排序;
*HashSet可以存储Null,TreeSet不允许存储Null(因为需要compareTo函数,Null值无法比较,所以会抛空指针异常)
*HashSet是基于HashMap实现的,TreeSet是基于TreeMap实现的
HashSet基本源码及相关继承:
TreeSet基本源码及相关继承:
TreeSet,add null时源码跟踪:
HashSet及TreeSet的基本遍历:
输出结果为:
二、 Map
以下等多种map的实现类,主要有一下共同点就是,实现了Map接口,并且都是通过Key-Value键值对形式存储数据。接下来单独说一下他们各自的特点
1、 Hashtable
首先Hashtable继承于Dictionary,实现了Map、Cloneable、Serializable接口,并且其中大部分方法都是通过synchronized关键字修饰意味着,Hashtable是线程同步的。
并且Hashtable的key-value值都不可以存储null值因为在put函数中,校验了value如果为null直接抛空指针,并且需要计算key值的hashcode值,所以hashtable是不允许存储null的,并且其实Hashtable也是通过Entry<K, V>对象的数组来进行数据存储的:
2、 HashMap
首先HashMap继承于AbstractMap,实现了Map、Cloneable、Serializable接口,但是hashMap并不是线程同步的,当然从另一种角度来说,HashMap的效率就要高于Hashtable了。
其次HashMap并不禁止key-value值存储null值,因为其实Map也是通过node节点进行参数存储的,如下代码:
另外,HashMap的存储方式其实类似excel一样的空间,在HashMap的存储空间中就是一列列小格子,每一列都代表着一个hash值,我们在将参数存储至HashMap后,在我们需要查找的时候,我们会先根据key值的hash值快速定位出那一列,然后通过equals方法快速定位出我们需要的键值对。另外需要补充说明的一点是HashMap使用的是散列算法,会尽量分散key值得hash值范围,以此保证尽量在‘同一列’不会保存过多的键值对。
源码有兴趣可以跟一下HashMap的get方法,最底层是一个find的递归函数:
3、 TreeMap
首先TreeMap继承于AbstractMap,实现了NavigableMap、Cloneable、Serializable接口,但是TreeMap并不是线程同步的,但是TreeMap是有序的。
另外需要说明一下的是TreeMap的key值是不允许存储null的,因为在TreeMap的存储过程中是需要通过key值调用compare方法来进行一个比较排序的,所以在TreeMap存储的过程中如果key值为null是会报空指针异常的。
还有一点要注意的是TreeMap和HashTable一样是通过Entry<K, V>对象来存储数据的,但是和Hashtable不一样的是,TreeMap的存储方式更接近与一个树装链表一样的存在,在存储一个Entry节点的过程中还会将这个Entry节点的左右及父节点一起保存下来。
如图,PS:如果需要更详细的了解TreeMap的相关实现方式可以查询一下红黑树,因为TreeMap就是基于红黑树(Red-Black tree)实现的。
4、 WeakHashMap
基本功能与HashMap相同,与HashMap最核心的不同即WeakHashMap是一个弱键值对,意思大致解释说当WeakHashMap的key值失去引用时,当发生了java的垃圾回收时,WeakHashMap的对应键值对会被清除:
5、 简单展示一些HashMap、Hashtable的遍历方式
a、 HashMap的遍历:
HashMap与TreeMap、WeakHashMap一样都有以下四种遍历方式:
输出结果如下:
PS: HashMap的keySet也可以通过iterator方式遍历就不在此展示了。
b、 Hashtable的遍历:
Hashtable除了和HashMap同样的四种遍历方式外,还有一种相对独有的遍历方式接下来进行展示一下:
代码依然是照片,手敲有益身心健康,没事还是可以多读一读jdk源码有很多值得学习的地方
Java集合简单解析的更多相关文章
- java集合List解析
作为一个Developer,Java集合类是我们在工作中运用最多的.最频繁的类.相比于数组(Array)来说,集合类的长度可变,更加适合于现代开发需求: Java集合就像一个容器,可以存储任何类型的数 ...
- Java集合简单介绍
再最前面分享一下我再学习集合时的方法: 1.首先了解各集合的定义和特点 2.集合的构造方法和常用方法(增删改查等) 3.了解集合使用的场景,再什么情况下使用什么类型的集合(关键是集合的特性) 4.了解 ...
- java集合简单特性
- Java 集合系列05之 LinkedList详细介绍(源码解析)和使用示例
概要 前面,我们已经学习了ArrayList,并了解了fail-fast机制.这一章我们接着学习List的实现类——LinkedList.和学习ArrayList一样,接下来呢,我们先对Linked ...
- Java 集合系列07之 Stack详细介绍(源码解析)和使用示例
概要 学完Vector了之后,接下来我们开始学习Stack.Stack很简单,它继承于Vector.学习方式还是和之前一样,先对Stack有个整体认识,然后再学习它的源码:最后再通过实例来学会使用它. ...
- Java 集合系列10之 HashMap详细介绍(源码解析)和使用示例
概要 这一章,我们对HashMap进行学习.我们先对HashMap有个整体认识,然后再学习它的源码,最后再通过实例来学会使用HashMap.内容包括:第1部分 HashMap介绍第2部分 HashMa ...
- Java 集合系列11之 Hashtable详细介绍(源码解析)和使用示例
概要 前一章,我们学习了HashMap.这一章,我们对Hashtable进行学习.我们先对Hashtable有个整体认识,然后再学习它的源码,最后再通过实例来学会使用Hashtable.第1部分 Ha ...
- java集合框架之java HashMap代码解析
java集合框架之java HashMap代码解析 文章Java集合框架综述后,具体集合类的代码,首先以既熟悉又陌生的HashMap开始. 源自http://www.codeceo.com/arti ...
- Java 集合系列 09 HashMap详细介绍(源码解析)和使用示例
java 集合系列目录: Java 集合系列 01 总体框架 Java 集合系列 02 Collection架构 Java 集合系列 03 ArrayList详细介绍(源码解析)和使用示例 Java ...
随机推荐
- Spring MVC 的 multipartResolver 不能同iWebOffice2006 共同使用
转:http://jamesby.iteye.com/blog/57381 项目使用iWebOffice2006,本来可以正常使用,但是系统有文件上传需求,故定义了一个multipartResolve ...
- win10 exe如何添加或禁用开机自启动项
一.添加开机自启动 1,先打开存放自启动软件文件的文件夹 方法①:在文件搜索框中输入或粘贴以下地址: C:\ProgramData\Microsoft\Windows\Start Menu\Progr ...
- Linux任务计划及周期性任务执行:at、crontab命令
一.概述 未来的某时间点执行一次某任务:at, batch 周期性运行某任务:crontab 这两个任务的执行结果:会通过邮件发送给用户 (本地终端用户之间的邮件通知) centos 5,6,7默认开 ...
- confluence 附件docx文件 乱码处理
服务器安装字体库 Fontconfig是一个用于配置和自定义字体访问的库 yum -y install fontconfig 拷贝需要的字体文件 fonts.zip(或自己电脑中的字体文件c:/Win ...
- DVWA--File Inclusion(不能远程包含的问题解决)
然后别以为这样就完了 我被这样坑了一下午 找到你对应版本的php 进去Ctrl+f 搜索url_allow——fopen 和include
- C++入门经典-例5.16-输出引用
1:如不加特殊说明,一般认为引用指的都是左值引用.引用实际上是一种隐式指针,它为对象建立一个别名,通过操作符&来实现,引用的形式如下: 数据类型 & 表达式: 例如: int a=10 ...
- Maven项目解决Remove '@override' annotation终极方案
特别提示:本人博客部分有参考网络其他博客,但均是本人亲手编写过并验证通过.如发现博客有错误,请及时提出以免误导其他人,谢谢!欢迎转载,但记得标明文章出处:http://www.cnblogs.com/ ...
- Rust Linking With C Library Functions
#[link(name = "c")] extern "C" { fn scanf(format: *const u8, ...); } fn main() { ...
- 在electron中使用sqlite:sql.js简介
在electron中使用sqlite:sql.js简介 在开发electron应用的时候如果想要使用sqlite3,步骤上除了npm安装以外还要rebuild,比较麻烦.如果你想找一个开箱即用的sql ...
- erlang实现排列组合问题
今天在公司做一个日志分析的任务,在做统计的时候,遇到这样一个问题, 之前已经将数据拆分好,出现这样一张中间表Table,简略写如下: A属性 B属性 C属性 D属性 1 3 ...