美文网首页
剖析Java 集合框架第一弹

剖析Java 集合框架第一弹

作者: 李元霸抢貂蝉 | 来源:发表于2020-11-22 21:36 被阅读0次

    序言

        写文章,这个想法我觉得很多人都有,我也很早就有,但是一直拖着,基于各种考虑:文笔不好啦,没有素材啦,没有深度啦。不管怎样,今天终于迈出了第一步,值得庆祝。
        主要分析生活中的感悟和自己专业技术(java方向),前期可能是以技术文章为主,毕竟对于新手来说,写生活感悟不比高考写命题作文简单。
        希望能分享我的所学所感,促使自己热爱生活,稳步提升。经常加班到深夜,走在寂静无人的路上,想的最多的就是活着是为了什么,这么拼,最后得到什么,为了什么。现在还没有确切答案,但是我希望我能热爱生活!
    话不多说,进入今天的分享主题,java集合框架

    1. 集合概况

    下图展示了Java容器类库关系图 image

    数据结构本质就是数组链表

    无序性是指存储的数据在底层数组中并非按照数组索引的顺序添加 ,而是根据数据的哈希值决定的。
    不可重复性是指添加的元素按照 equals()判断时 ,返回 false,需要同时重写 equals()方法和 HashCode()方法。

    Java容器类类库的用途是"保存对象",并将其划分为两个不同的概念:
    Collecton 一个独立元素的序列

    • List 必须按照插入的顺序保存元素

    • Arraylist:基于动态数组实现,支持随机访问、有序、重复、线程不安全、扩容为当前的1.5倍。

    • Vector:和 ArrayList 类似,线程安全,扩容为当前的2倍。

    • LinkedList:双向链表,只能顺序访问,快速插入和删除元素。还可以用作栈、队列和双向队列。线程不安全。

    • Set 不能有重复元素

    • HashSet(无序,唯一): 基于哈希表实现,底层采用 HashMap 来保存元素

    • TreeSet(有序,唯一):红黑树(自平衡的排序二叉树),支持有序性操作。但是查找效率不如 HashSet,HashSet 查找的时间复杂度为 O(1),TreeSet 则为 O(logN)。

    • LinkedHashSet:是 HashSet 的子类,内部使用双向链表维护元素的插入顺序,顺序为插入顺序或者最近最少使用(LRU)顺序。

    • Queue 按照排队规则来确定对象产生的顺序

    • LinkedList:可以用它来实现双向队列和栈。

    • PriorityQueue:基于堆结构实现,可以用它来实现优先队列。Map. 一组成对的"键值对"对象,允许你使用键来查找值。

    • HashMap:JDK1.8 由数组+链表组成的,当链表长度大于阈值(默认为 8)(当前数组的长度大于 64)时,将链表转化为红黑树,以减少搜索时间,key/value可为null

    • LinkedHashMap:继承自 HashMap,使用双向链表来维护元素的顺序,顺序为插入顺序或者最近最少使用(LRU)顺序。

    • TreeMap:红黑树

    • Hashtable:数组+链表组成的,线程安全的,它是遗留类,不应该去使用它,而是使用 ConcurrentHashMap 来支持线程安全,key/value不可为null

    线程安全的集合类

    ArraylistLinkedListHashMapHashSetTreeSetTreeMapPriorityQueue 都不是线程安全的。java.util.concurrent 包中提供了很多并发容器供你使用:

    • ConcurrentHashMap: 可以看作是线程安全的 HashMap
    • CopyOnWriteArrayList:可以看作是线程安全的 ArrayList,在读多写少的场合性能非常好,远远好于 Vector.
    • ConcurrentLinkedQueue:高效的并发队列,使用链表实现。可以看做一个线程安全的 LinkedList,这是一个非阻塞队列。
    • BlockingQueue: 这是一个接口,JDK 内部通过链表、数组等方式实现了这个接口。表示阻塞队列,非常适合用于作为数据共享的通道。
    • ConcurrentSkipListMap :跳表的实现。这是一个Map,使用跳表的数据结构进行快速查找。

    2. 如何选用集合

    以下基于《Java编程思想》第四版第17章 容器深入研究 总结

    • List的选择:
      最佳的做法可能是将ArrayList作为默认首选,只有需要额外的功能,或者频繁的进行插入和删除而使性能变差的时候,才去选择LinkedList。如果使用的是固定数量的元素,可以Arrays.asList()产生的列表List。或者直接使用数组。应该避免使用Vector,应该使用CopyOnWriteArrayList,专门用于并发编程。
    • Set的选择
      HashSet的性能最好。只有需要一个有序的Set时,才应该使用 TreeSet。对于插入操作, LinkedHashSetHashSet的代价更高,这是由维护链表所带来额外。
    • Map的选择
      SetHashMap的性能最好。需要排序时使用TreeMap,LinkedHashMap在插入时比HashMap慢一点。避免使用HashTable,使用 ConcurrentHashMap 来支持线程安全。

    3. 集合容器中的设计模式

    迭代器模式

    Collection 继承了 Iterable 接口,其中的 iterator() 方法能够产生一个 Iterator 对象,通过这个对象就可以迭代遍历 Collection 中的元素。

    public interface Iterator<E> {
        //集合中是否还有元素
        boolean hasNext();
        //获得集合中的下一个元素
        E next();
        ......
    }
    

    从 JDK 1.5 之后可以使用 foreach 方法来遍历实现了 Iterable 接口的聚合对象。

    List<String> list = new ArrayList<>();
    for (String item : list) {
        System.out.println(item);
    }
    

    适配器模式

    java.util.Arrays#asList() 可以把数组类型转换为 List 类型。

    @SafeVarargs
    public static <T> List<T> asList(T... a)
    
    Integer[] arr = {1, 2, 3};
    List list = Arrays.asList(arr);
    //也可以使用以下方式调用 asList():
    List list = Arrays.asList(1, 2, 3);
    

    应该注意的是 asList() 的参数为泛型的可变长参数,不能使用基本类型数组作为参数,只能使用相应的包装类型数组。

    感谢看官看完!本来想挨个分析源码,还是太年轻。且不说花了多长时间,限于篇幅,码字,排版,编辑...真是令人头大

    磕磕绊绊写完了基本篇,下篇源码分析,敬请期待!

    ps

    集合框架脑图 image

    相关文章

      网友评论

          本文标题:剖析Java 集合框架第一弹

          本文链接:https://www.haomeiwen.com/subject/dhjpiktx.html