可能很多人会问Java集合真的那么重要吗,就像为什么现在很多企业都喜欢先问算法一样,目的就是考察你对Java基础掌握的程度如何。下面我将列出了一些关于Java集合的重要问题,让我们一起看看3到5年的Java开发工程师是如何回答这些问题的。
比较/集合 | List | Set | Map |
元素 | 可以重复 | 不可重复(equals比较) | 键必须唯一,但是值可以重复 |
顺序 | 有序 | 无序(有hashcode决定) | 无序 |
常用方法 | add( )、remove( )、clear( )、get( )、contains( )、size( ) | add( )、remove( )、clear( )、contains( )、size( ) | put( )、get( )、remove( )、clear( )、containsKey( )、containsValue( )、keySet( )、values( )、size( ) |
常见实现类 | ArrayList、LinkedList、Vector | HashSet、LinkedHashSet、TreeSet | HashMap、HashTable |
允许插入null | 可以插入多个null | 只允许插入一个null | 键只允许插入一个null,值可以插入多个null值 |
1.堆栈:是一种先进后出的数据结构(容器),就像弹夹一样(压子弹)
2.队列:先进先出(当多个任务分配给打印机时,为了防止冲突,创建一个队列,把任务入队,按先入先出的原则处理任务。当多个用户要访问远程服务端的文件时,也用到队列,满足先来先服务的原则。)
原来这才是使用LinkedList实现堆栈和队列的方式!
1.当遍历数组结构的集合时用for或者foreach都行,在固定长度或长度不需要计算的时候for循环效率高于foreach。
2.在不确定长度或者计算长度有损性能的时候用foreach比较方便。
3.对于数组来说,for和foreach循环效率差不多,但是对于链表来说,for循环效率明显比foreach低。
如果使用foreach,当数据量大的时候有可能会导致系统崩溃,为什么会很糟糕,因为for循环时要获取第i个元素必须从头开始遍历,而iterator遍历就是从头开始遍历,遍历完只需要一次(foreach循环就是用的iterator)。
map.entrySet即可,entrySet的方式整体都是比keySet方式要高一些,代码如下:
public static void main(String[] args) {
Map map=new TreeMap();
map.put(11, "张三");
map.put(22, "李四");
map.put(33, "王五");
map.put(11, "张三三");
for(Map.Entry entry:map.entrySet()){
System.out.println("key:"+entry.getKey()+"\t"+"value:"+entry.getValue());
}
}
如果就只获取key来说,两者的差别并不大,但是如果要获取value,还是entrySet的效率会更好。因为keySet需要从map中再次根据key获取value(keySet相当于遍历了2次),而entrySet是一次都全部获取出来。
map.get(key)获取的时候,底层其实根据key的hashcode值经过哈希算法得到一个hash值然后作为索引映射到对应table数组的索引位置,这是一次密集型计算,很耗费CPU,如果有大量的元素,则会使CPU使用率飙升,影响响应速度,而entrySet()返回的set里边元素都是Map.Entry类型,key和value就是这个类的一个属性,entry.getKey()和entry.getValue()效率肯定很高。
HashMap | HashSet |
实现了Map接口 | 实现了Set接口 |
储存键值对 | 只存储对象 |
使用put()方法将元素放入map中 | 使用add()方法将元素放入set中 |
使用键对象来计算hashcode值 | HashSet使用成员对象来计算hashcode值,对于两个对象来说hashcode可能相同,所以equals()方法用来判断对象的相等性,如果两个对象不同的话,那么返回false。 |
HashMap比较快,因为是使用唯一的键来获取对象 | HashSet较HashMap来说比较慢 |
1.HashSet是基于Hash算法实现的,其性能通常都优于TreeSet。为快速查找而设计的Set,我们通常都应该使用HashSet,在我们需要排序的功能时,我们才使用TreeSet。
2.HashSet还有一个子类LinkedHashSet,LinkedHashSet集合也是根据元素hashCode值来决定元素存储位置,但它同时使用链表维护元素的次序,当遍历该集合时候,LinkedHashSet将会以元素的添加顺序访问集合的元素。 如果我们需要迭代遍历的顺序为插入顺序或者访问顺序,那么 LinkedHashSet 是需要我们首先考虑的。
HashMap中的数据结构是数组+单链表的组合,以键值对(key-value)的形式存储元素的,通过put()和get()方法储存和获取对象。可以理解为两列多行的表格,第一列中存储索引或者起标识作用的对象,第二列存储我们实际要用的对象,当我们需要第二列中某个对象时,就去找这个对象的索引。
1.将对象放到Set集合或者是想作为Map的key时,那么你必须重写equals()方法,这样才能保证唯一性。hashCode 和 equals的关系:两个对象 equals的时候,hashCode必须相等,但hashCode相等,对象不一定equals。
2.必须重写hashCode()的情况:如果你的对象想放进散列存储的集合中(比如:HashSet,LinkedHashSet)或者想作为散列Map(例如:HashMap,LinkedHashMap等等)的Key时,在重写equals()方法的同时,必须重写hashCode()方法。hashCode()方法存在的主要目的就是提高效率,但是如果你想把对象放到散列存储结构的集合中时,是必须要重写的。
hashCode并不能表现其唯一性,但是有离散性,其意义就是类似于进行hashMap等操作时,加快对象比较的速度,进而加快对象搜索的速度。
1.当Map中插入、删除和定位元素这类操作时,HashMap是最好的选择。
2.如果你需要对一个有序的key集合进行遍历,TreeMap是更好的选择。基于你的collection的大小,向HashMap中添加元素会更快,将map换为TreeMap可以进行有序key的遍历。
Collection 是一个集合根接口,Collections 是一个包装类(工具类/帮助类)。它包含有各种有关集合操作的静态多态方法。此类不能实例化,就像一个工具类,用于对集合中元素进行排序、搜索以及线程安全等各种操作,服务于Java的Collection框架。
两个都有用于自然排序(使用Comparable)或基于标准的排序(使用Comparator)的重载方法sort()。Collections内部使用数组排序方法,所有它们两者都有相同的性能,只是Collections需要花时间将列表转换为数组。
Comparable | 用来提供对象的自然排序,我们可以使用它来提供基于单个逻辑的排序。 |
Comparator | 提供不同的排序算法,我们可以选择需要使用的Comparator来对给定的对象集合进行排序。 |
Comparable和Comparator接口被用来对对象集合或者数组进行排序。
--------------------------------------------保持专注,世界才会为你让路。--------------------------------------------