java容器集合类的区别用法_Java容器笔记(二):不同集合实现类的特点与区别...

package java.util包中的Collection相关接口和类如下图:

7cf84a39415f?utm_campaign=maleskine&utm_content=note&utm_medium=seo_notes&utm_source=recommendation

Collection.png

仅讨论Java.util包中的常见集合类,不涉及java.util的子包concurrent中的并发集合类。

可以这样简单的来对待容器中集合:

7cf84a39415f?utm_campaign=maleskine&utm_content=note&utm_medium=seo_notes&utm_source=recommendation

Collection_common.png

1、 List、Set、Queue三个接口的意义

首先,接口是一种规范,按照接口规范实现接口的的方法,就能提供所期望的功能。

三个接口只是规定了三类不同特性的集合,有时候实现类会同时实现其中的多个特性,如LinkedList:

7cf84a39415f?utm_campaign=maleskine&utm_content=note&utm_medium=seo_notes&utm_source=recommendation

LinkedList.png

LinkedList既实现了List接口,又(间接)实现了Queue接口。它是List还是Queue呢?所以要理解这三个接口存在的意义。

List:元素有序,元素可重复,添加的元素放在最后(按照插入顺序保存元素)

//Appends the specified element to the end of this list

boolean add(E e);

List是有序的Collection,使用此接口能够精确的控制每个元素插入的位置。用户能够使用索引(元素在List中的位置,类似于数组下标)来访问List中的元素,类似于Java的数组。List允许有相同的元素。除了具有Collection接口必备的iterator()方法外,List还提供一个listIterator()方法,返回一个 ListIterator接口,和标准的Iterator接口相比,ListIterator多了一些add()之类的方法,允许添加,删除,设定元素, 还能向前或向后进行双向遍历。

**Set:元素无序并且不允许重复元素 **

// If this set already contains the element, the call leaves the set

//unchanged and returns false. In combination with the

//restriction on constructors, this ensures that sets never contain

//duplicate elements.

boolean add(E e);

Set是一种不包含重复的元素的Collection,即任意的两个元素e1和e2都有e1.equals(e2)=false,Set最多有一个null元素。

很明显,Set的构造函数有一个约束条件,传入的Collection参数不能包含重复的元素。

请注意:必须小心操作可变对象(Mutable Object)。如果一个Set中的可变元素改变了自身状态导致Object.equals(Object)=true将导致一些问题。

Queue:元素有序,先进先出

它们都是通过扩展Collection接口而来。Collection接口规范了存放一组对象的方式,而这组对象在存放时具有哪些特点(元素有序还是无序,元素允不允许重复等),则通过这三个接口进一步规范,实现了某个接口就具有某个接口规定的特性。而具体的实现方式的不同,使得各实现类的应用场景不同。

2、各实现类的特点

先看List:

ArrayList

数据结构为数组,访问快(可以直接通过下标访问),增删慢,未实现线程同步。

ArrayList实现了可变大小的数组。它允许所有元素,包括null。ArrayList没有同步。size,isEmpty,get,set方法运行时间为常数。但是add方法开销为分摊的常数,添加n个元素需要O(n)的时间。其他的方法运行时间为线性。

每个ArrayList实例都有一个容量(Capacity),即用于存储元素的数组的大小。这个容量可随着不断添加新元素而自动增加,但是增长算法 并没有定义。当需要插入大量元素时,在插入前可以调用ensureCapacity方法来增加ArrayList的容量以提高插入效率。和LinkedList一样,ArrayList也是非同步的(unsynchronized)。

LinkedList

数据结构为链表,增删速度快,查询慢,未实现线程同步

LinkedList实现了List接口,允许null元素。此外LinkedList提供额外的get,remove,insert方法在 LinkedList的首部或尾部。这些操作使LinkedList可被用作堆栈(stack),队列(queue)或双向队列(deque)。

Vector类

数据结构为数组,访问快(可以直接通过下标访问),增删慢,实现线程同步

Vector非常类似ArrayList,但是Vector是同步的。由Vector创建的Iterator,虽然和 ArrayList创建的Iterator是同一接口,但是,因为Vector是同步的,当一个Iterator被创建而且正在被使用,另一个线程改变了 Vector的状态(例如,添加或删除了一些元素),这时调用Iterator的方法时将抛出 ConcurrentModificationException。

再看Set

HashSet

数据结构为哈希表,元素无序、不重复,至多有一个null元素

底层使用 HashMap 来保存所有元素,因此 HashSet 的实现比较简单,相关 HashSet 的操作,基本上都是直接调用底层 HashMap 的相关方法来完成。

特点如下

 不能保证元素的排列顺序,顺序有可能发生变化

 不是同步的

 集合元素可以是null,但只能放入一个null

当向HashSet结合中存入一个元素时,HashSet会调用该对象的hashCode()方法来得到该对象的hashCode值,然后根据 hashCode值来决定该对象在HashSet中存储位置。

简单的说,HashSet集合判断两个元素相等的标准是两个对象通过equals方法比较相等,并且两个对象的hashCode()方法返回值相 等

注意,如果要把一个对象放入HashSet中,重写该对象对应类的equals方法,也应该重写其hashCode()方法。其规则是如果两个对 象通过equals方法比较返回true时,其hashCode也应该相同。另外,对象中用作equals比较标准的属性,都应该用来计算 hashCode的值。

LinkedHashSet

数据结构是哈希表和链表,与HashSet相比访问更快,插入时性能稍微

LinkedHashSet继承自HashSet。同样是根据元素的hashCode值来决定元素的存储位置,但是它同时使用链表维护元素的次序。这样使得元素看起 来像是以插入顺序保存的,也就是说,当遍历该集合时候,LinkedHashSet将会以元素的添加顺序访问集合的元素。

LinkedHashSet在迭代访问Set中的全部元素时,性能比HashSet好,但是插入时性能稍微逊色于HashSet。

TreeSet

数据结构是二叉树(红黑树),元素可排序、不重复

TreeSet是SortedSet接口的唯一实现类,TreeSet可以确保集合元素处于排序状态。TreeSet支持两种排序方式:自然排序和定制排序,其中自然排序为默认的排序方式。向TreeSet中加入的应该是同一个类的对象。

TreeSet判断两个对象不相等的方式是两个对象通过equals方法返回false,或者通过CompareTo方法比较没有返回0

自然排序

(1)TreeSet内的元素实现Comparable接口,重写该接口的compareTo(Object obj)方法,以此确定排序。(元素必须实现该接口,否则程序会抛出异常)。

(2)当重写元素对应类的equals()方法时,应该保证该方法与compareTo(Object obj)方法有一致的结果,即如果两个对象通过equals()方法比较返回true时,这两个对象通过compareTo(Object obj)方法比较结果应该也为0(即相等)

定制排序--

自然排序是根据集合元素的大小,以升序排列,如果要定制排序,应该使用Comparator接口,实现int compare(T o1,T o2)方法

个人看法:两种方式都是为了排序,一种是元素本身实现`Comparable`接口,另一种则是当元素本身没有实现`Comparable`接口时,可以通过`Comparator`接口(传入构造器`TreeSet(Comparator comparator)`),两者本身没有什么区别。

Java : Comparable vs Comparator

In short, there isn't much difference. They are both ends to similar means. In general implement comparable for natural order, (natural order definition is obviously open to interpretation), and write a comparator for other sorting or comparison needs.

最后看Queue

ArrayDeque

数据结构为数组,双端队列,在队头队尾均可心插入或删除元素

实现了DeQueue接口。DeQueue(Double-ended queue)继承了Queue接口,创建双向队列,灵活性更强,可以前向或后向迭代,

7cf84a39415f?utm_campaign=maleskine&utm_content=note&utm_medium=seo_notes&utm_source=recommendation

ArrayDeque.png

PriorityQueue

数据结构为优先级队列,元素不允许null,非同步

优先级队列是一种什么样的数据结构

优先级队列是不同于先进先出队列的另一种队列。每次从队列中取出的是具有最高优先权的元素。

(1)优先级队列不是同步的(线程安全版本为PriorityBlockingQueue)

队列的获取操作如poll(),peek()和element()是访问的队列的头,保证获取的是最小的元素(根据指定的排序规则)

(2)返回的迭代器并不保证提供任何的有序性

(3)优先级队列不允许null元素,否则抛出NullPointException。

3、有助于理解的问答

遍历一个List有哪些不同的方式?

List strList = new ArrayList<>();

//使用for-each循环

for(String obj : strList){

System.out.println(obj);

}

//using iterator

Iterator it = strList.iterator();

while(it.hasNext()){

String obj = it.next();

System.out.println(obj);

}

使用迭代器更加线程安全,因为它可以确保,在当前遍历的集合元素被更改的时候,它会抛出ConcurrentModificationException。

在迭代一个集合的时候,如何避免ConcurrentModificationException?

在遍历一个集合的时候,我们可以使用并发集合类来避免ConcurrentModificationException,比如使用CopyOnWriteArrayList,而不是ArrayList。

4、常见面试题

List和Set的区别

List有序,允许重复,Set无序,不允许重复

ArrayList与LinkedList的区别

ArrayList为数组结构,LinkedList为链表结构。所以,一个访问快,一个增删快。均未实现线程同步或者说都是线程不安全的。

(1)ArrayList是由Array所支持的基于一个索引的数据结构,所以它提供对元素的随机访问,复杂度为O(1),但LinkedList存储一系列的节点数据,每个节点都与前一个和下一个节点相连接。所以,尽管有使用索引获取元素的方法,内部实现是从起始点开始遍历,遍历到索引的节点然后返回元素,时间复杂度为O(n),比ArrayList要慢。

(2)与ArrayList相比,在LinkedList中插入、添加和删除一个元素会更快,因为在一个元素被插入到中间的时候,不会涉及改变数组的大小,或更新索引。

(3)LinkedList比ArrayList消耗更多的内存,因为LinkedList中的每个节点存储了前后节点的引用。

ArrayList与Vector的区别

Vector同步,ArrayList不同步。

ArrayList和Vector在很多时候都很类似。

(1)两者都是基于索引的,内部由一个数组支持。

(2)两者维护插入的顺序,我们可以根据插入顺序来获取元素。

(3)ArrayList和Vector的迭代器实现都是fail-fast的。

(4)ArrayList和Vector两者允许null值,也可以使用索引值对元素进行随机访问。

以下是ArrayList和Vector的不同点。

(1)Vector是同步的,而ArrayList不是。然而,如果你寻求在迭代的时候对列表进行改变,你应该使用CopyOnWriteArrayList。

(2)ArrayList比Vector快,它因为有同步,不会过载。

(3)ArrayList更加通用,因为我们可以使用Collections工具类轻易地获取同步列表和只读列表。

Array和ArrayList有何区别?什么时候更适合用Array?

Array可以容纳基本类型和对象,而ArrayList只能容纳对象。

Array是指定大小的,而ArrayList大小是不固定的。

Array没有提供ArrayList那么多功能,比如addAll、removeAll和iterator等。尽管ArrayList明显是更好的选择,但也有些时候Array比较好用。

(1)如果列表的大小已经指定,大部分情况下是存储和遍历它们。

(2)对于遍历基本数据类型,尽管Collections使用自动装箱来减轻编码任务,在指定大小的基本类型的列表上工作也会变得很慢。

(3)如果你要使用多维数组,使用[][]比List>更容易。

5、简单总结

如果元素唯一(不允许重复),使用Set:

支持排序则选择TreeSet,不支持排序选择HashSet。

如果元素不唯一(允许重复),使用List:

查询多则选择ArrayList,增删多则选择:LinkedList。

Vector虽然是ArrayList的线程同步版本,但还有更好地选择。

关于Queue,由于工作中从未接触过,待补充。

参考

未一一列出,待补充

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/409083.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

[机器学习笔记]Note5--归一化

继续是机器学习课程的笔记&#xff0c;这节课会介绍归一化的内容。 过拟合问题 这节课会介绍一个在机器学习过程中经常会遇到的问题–过拟合。通常&#xff0c;当我们有非常多的特征&#xff0c;我们可以学习得到的假设可能非常好地适应训练集&#xff0c;即代价函数可能几乎…

combobox的联动练习

老师的项目中&#xff0c;网站右上有四个联动的combobox&#xff0c;今天第一次尝试解决。外观如图&#xff1a; 首先第一次登陆系统时&#xff0c;需要分别载入4个框中的数据。通过easyui-combobox的url 属性请求php返回json数据即可&#xff0c;json中的value是名称&#xff…

[机器学习笔记]Note6--神经网络:表达

继续是机器学习课程的笔记&#xff0c;这节课会介绍神经网络的内容。 非线性假设 在之前的课程中&#xff0c;我们看到使用非线性的多项式能够帮助我们建立更好的分类模型。假设我们有非常多的特征&#xff0c;例如100个变量&#xff0c;我们希望用这100个特征来构建一个非线…

[机器学习笔记]Note7--神经网络:学习

继续是机器学习课程的笔记&#xff0c;这节课会继续介绍神经网络的内容&#xff0c;上一节主要是基本的介绍&#xff0c;以及模型表示&#xff0c;而本节会介绍代价函数&#xff0c;反向传播算法等。 神经网络代价函数 首先是有如下一个神经网络&#xff0c;这里将首先介绍一些…

[机器学习笔记]Note8--机器学习应用建议

继续是机器学习课程的笔记&#xff0c;本节课的内容主要是一些机器学习应用的建议&#xff0c;包括对假设的评估&#xff0c;如何处理过拟合和欠拟合等。 觉得下一步做什么 到目前为止&#xff0c;我们已经学习了线性回归&#xff0c;逻辑回归以及神经网络&#xff0c;梯度下…

Unity3d 手机屏幕自动适配

现在&#xff0c;市场上的手机分辨率多样化。带给开放人员一个很大的“跨界问题”。本人&#xff0c;昨晚突发奇想。手机分辨率多样化&#xff0c;但手机开放人员&#xff0c;制作UI时&#xff0c;最为重要的两个因素就是Position(位置)、Scale(大小)&#xff1b; 我引入一个“…

[机器学习笔记]Note9--机器学习系统设计

继续是机器学习课程的笔记&#xff0c;本节课的内容主要是介绍如何设计一个机器学习系统。 首先要做什么 这节课将介绍如何设计一个机器学习系统&#xff0c;首先是以设计一个垃圾邮件分类器算法作为例子。 为了解决这个问题&#xff0c;首先要决定的是如何选择并表达特征向…

java 图片阴影_Java 为 PPT 中的图形添加阴影效果

在PowerPoint文档中&#xff0c;给图形添加阴影效果能增强图形的立体感&#xff0c;使其贴近现实效果&#xff0c;提升文档的美观度。 本文将展示如何使用Free Spire.Presentation for Java为PPT中的图形添加阴影效果。除了文中展示的预设阴影效果&#xff0c;还可以添加内部阴…

nyoj 21 三个水杯 BFS

三个水杯 时间限制&#xff1a;1000 ms | 内存限制&#xff1a;65535 KB难度&#xff1a;4描述给出三个水杯&#xff0c;大小不一&#xff0c;并且只有最大的水杯的水是装满的&#xff0c;其余两个为空杯子。三个水杯之间相互倒水&#xff0c;并且水杯没有标识&#xff0c;只…

论文阅读(1)--Fine-grained Image Classification by Exploring Bipartite-Graph Labels

这是阅读《Fine-grained Image Classification by Exploring Bipartite-Graph Labels》这篇论文所做的笔记。 这篇论文是来自NEC实验室&#xff0c;是一篇有关细粒度分类/精细分类方面的论文。 0. 摘要 首先提出一个问题&#xff0c;给定一张食物的图片&#xff0c;对于一个…

java注释和注解_深入理解JAVA注解(Annotation)以及自定义注解

Java 注解(Annotation)又称 Java 标注&#xff0c;是 JDK5.0 引入的一种注释机制。Java 语言中的类、方法、变量、参数和包等都可以被标注。注解可以看作是一种特殊的标记&#xff0c;在程序在编译或者运行时可以检测到这些标记而进行一些特殊的处理。本文对 Annotation 进行了…

[机器学习笔记]Note10--支持向量机(SVM)

继续是机器学习课程的笔记&#xff0c;这节课的内容是介绍支持向量机&#xff08;SVM&#xff09;的内容。SVM是一个非常强大且流行的算法&#xff0c;在一些情况下&#xff0c;面对一些复杂的非线性问题可以提供比逻辑回归或神经网络更加简洁更加有效的结果。 优化目标 首先…

[机器学习笔记]Note11--聚类

继续是机器学习课程的笔记&#xff0c;本节介绍的是聚类方法&#xff0c;主要是K-means算法。 非监督学习 首先介绍监督学习和非监督学习的区别。一个监督学习的例子如下图&#xff0c;给定训练集如:{(x(1),y(1)),(x(2),y(2)),…,(x(m),y(m))},每个训练实例都有对应的标签。 …