当前位置:   article > 正文

java_container_交换3p性操

交换3p性操

集合概述

Java集合概览

集合由两个接口派生而来,Collection:存放单一元素;Map:存放键值对。Collection下三个子接口:List、Set、Queue

img

List、Set、Queue、Map

  • List:有序,按对象进入顺序保存对象,可重复,允许多个null元素对象,可以使用Iterator取出所有元素逐一遍历,还可以使用get(int index)获取指定下表的对象

    • ArrayList:ArrayList是List的主要实现类,底层使用Object[]存储,适用于频繁的查找工作,线程不安全
    • Vector:是List的古老实现类,底层使用Object[]存储,线程安全。
    • LinkedList:使用双向链表数据结构(1.6前为循环链表,1.7取消了循环)
  • Set:无序不可重复,最多允许一个null元素对象,只能用Iterator接口取得所有元素,再逐一遍历

    • HashSet:基于HashMap实现,支持快速查找,不支持有序性操作。并且失去了元素的插入顺序信息,即使用Iterator遍历HashSet得到的结果不确定。
    • LinkedHashSet:LinkedHashSet具有HashSet的查找效率,内部使用双向链表维护元素插入顺序(FIFO)。
    • TreeSet:基于红黑树实现,支持有序性操作,如根据一个范围查找元素的操作。效率不如HashSetHashSet时间复杂度为O(1),TreeSet为O(logN)
  • Queue:先后顺序有序可重复

    • PriorityQueue:Object[]实现二叉堆
    • ArrayQueue:Object[]+双指针
  • Map :键值对存储,key无序不可重复,value无序可重复。每个键最多一个值

    • HashMap: JDK1.8 之前 HashMap 由数组+链表组成的,数组是 HashMap 的主体,链表则是主要为了解决哈希冲突而存在的(“拉链法”解决冲突)。JDK1.8 以后在解决哈希冲突时有了较大的变化,当链表长度大于阈值(默认为 8)(将链表转换成红黑树前会判断,如果当前数组的长度小于 64,那么会选择先进行数组扩容,而不是转换为红黑树)时,将链表转化为红黑树,以减少搜索时间
    • LinkedHashMapLinkedHashMap 继承自 HashMap,所以它的底层仍然是基于拉链式散列结构即由数组和链表或红黑树组成。另外,LinkedHashMap 在上面结构的基础上,增加了一条双向链表,使得上面的结构可以保持键值对的插入顺序。同时通过对链表进行相应的操作,实现了访问顺序相关逻辑。详细可以查看:《LinkedHashMap 源码详细分析(JDK1.8)》 (opens new window)
    • Hashtable: 数组+链表组成的,数组是 Hashtable 的主体,链表则是主要为了解决哈希冲突而存在的
    • TreeMap: 红黑树(自平衡的排序二叉树)

如何选用集合

主要根据集合的特点来选用,比如我们需要根据键值获取到元素值时就选用 Map 接口下的集合,需要排序时选择 TreeMap,不需要排序时就选择 HashMap,需要保证线程安全就选用 ConcurrentHashMap

当我们只需要存放元素值时,就选择实现Collection 接口的集合,需要保证元素唯一时选择实现 Set 接口的集合比如 TreeSetHashSet,不需要就选择实现 List 接口的比如 ArrayListLinkedList,然后再根据实现这些接口的集合的特点来选用。

为什么要使用集合

使用数组存储对象有一定的弊端。数组一旦声明长度不可变、类型不可变、有序、可重复,特点单一。集合提高了数据存储的灵活性,还可以保存具有映射关系的数据。


List

ArrayList和LinkedList的区别

  • **是否保证线程安全:**A和L都是不同步的,不保证线程安全
  • **底层数据结构:**A使用Object[]数组,L使用双向链表数据结构(1.6前为循环链表,1.7取消了循环)
  • **插入和删除是否受元素位置影响:**A采用数组存储,受元素位置影响,第i和后面n-i个元素都要向前/后移,o(n-i);L需要移动到指定位置,再执行插入,o(n)
  • **是否支持快速随机访问:**L不支持,A支持。
  • **空间占用:**A体现在list列表结尾会预留一定的容量空间。L体现再每一个元素要消耗比A更多的空间(要存放直接后继、前驱以及数据)

Set

比较TreeSet、HashSet、LinkedHashSet的异同?

  • 三者都是Set接口的实现类,都能保证元素唯一,并且都不是线程安全的
  • 主要区别在于底层数据结构不同
  • 应用场景不同,HashSet不需要保证元素插入和取出顺序的场景,LinkedHashSet用于保证元素插入和取出顺序的场景,TreeSet用于支持对元素自定义排序规则的场景。

Queue

Queue和Deque的区别

  1. Queue
  • Q单端队列,只能从一段插入元素,另一端删除元素,遵循FIFO
  • Q扩展了Collection接口,根据因为容量问题导致操作失败后处理方式不同分为两类方法:
    Queue 接口抛出异常返回特殊值
    插入队尾add(E e)offer(E e)
    删除队首remove()poll()
    查询队首元素element()peek()
  1. Deque
  • D是双端队列,在队列两端均可插入/删除元素
  • 扩展了Queue的接口,增加了在队首队尾插入删除的操方法:
Deque 接口抛出异常返回特殊值
插入队首addFirst(E e)offerFirst(E e)
插入队尾addLast(E e)offerLast(E e)
删除队首removeFirst()pollFirst()
删除队尾removeLast()pollLast()
查询队首元素getFirst()peekFirst()
查询队尾元素getLast()peekLast()

ArrayDeque和LinkedList的区别

都实现了Deque接口,都具有队列功能

  • A是基于可变长的数组和双指针来实现,L通过链表实现
  • A不支持存储NULL数据,L支持
  • A在JDK1.6后才引入,L在1.2就存在
  • A插入可能存在扩容过程,均摊后插入操作依然为O(1),L不存在扩容,但每次插入需要申请新的堆空间,均摊性能相比更慢
  • A实现队列比L更好,A也可以用来实现栈

PriorityQueue

1.5中被引入,与Queue区别在于元素出队顺序与优先级有关

  • PriorityQueue 利用二叉堆数据结构来实现,底层使用变长的数据存储数据
  • PQ通过堆元素上浮下沉,实现logn时间复杂度内插入删除
  • PQ非线程安全,不支持存储NULLno-comparable的对象
  • PQ默认小顶堆,但可以接收一个Comparator作为构造参数,从而定义元素优先级先后。

Map

HashMap和HashTable的区别

  • **线程是否安全:**M非线程安全,T线程安全。T内部方法基本都是经过synchronized修饰。
  • **效率:**H更高,T基本被淘汰
  • **对Null key和Null value的支持:**M可以存储null的key和value,但null作为键只能有一个,null作为值可以有很多个;T不允许
  • 初始容量大小和每次扩容量大小的不同
    • **不指定容量初始值:**T默认11,每次扩容2n+1。M默认16,每次扩容,容量变为原来的两倍
    • **指定容量初始值:**T使用给定大小,M会将其扩充为2的幂次方大小(HashMap中的tableSizeFor()方法保证)。即M总会使用2的幂作为哈希表大小
  • **底层数据结构:**1.8以后M解决哈希冲突上有了较大的变化,当链表长度大于阈值(默认8)(将链表转换为红黑树前会判断,如果当前数组长度小于64,那么会选择先进行数组扩容,而不是转换为红黑树),将链表转换为红黑树,以减少搜索时间。T没有

HashMap中带有初始容量的构造函数:

    public HashMap(int initialCapacity, float loadFactor) {
        if (initialCapacity < 0)
            throw new IllegalArgumentException("Illegal initial capacity: " +
                                               initialCapacity);
        if (initialCapacity > MAXIMUM_CAPACITY)
            initialCapacity = MAXIMUM_CAPACITY;
        if (loadFactor <= 0 || Float.isNaN(loadFactor))
            throw new IllegalArgumentException("Illegal load factor: " +
                                               loadFactor);
        this.loadFactor = loadFactor;
        this.threshold = tableSizeFor(initialCapacity);
    }
     public HashMap(int initialCapacity) {
        this(initialCapacity, DEFAULT_LOAD_FACTOR);
    }
    /**
     * Returns a power of two size for the given target capacity.
     */
    static final int tableSizeFor(int cap) {
        int n = cap - 1;
        n |= n >>> 1;
        n |= n >>> 2;
        n |= n >>> 4;
        n |= n >>> 8;
        n |= n >>> 16;
        return (n < 0) ? 1 : (n >= MAXIMUM_CAPACITY) ? MAXIMUM_CAPACITY : n + 1;
    }
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27

HashMap和HashSet的区别

看过 HashSet 源码的话就应该知道:HashSet 底层就是基于 HashMap 实现的。(HashSet 的源码非常非常少,因为除了 clone()writeObject()readObject()HashSet 自己不得不实现之外,其他方法都是直接调用 HashMap 中的方法。

HashMapHashSet
实现了 Map 接口实现 Set 接口
存储键值对仅存储对象
调用 put()向 map 中添加元素调用 add()方法向 Set 中添加元素
HashMap 使用键(Key)计算 hashcodeHashSet 使用成员对象来计算hashcode 值,对于两个对象来说 hashcode 可能相同,所以equals()方法用来判断对象的相等性

HashMap和TreeMap的区别

都继承自AbstractMap,T还实现了NavigableMap接口和SortedMap接口

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-e8Jdmcdp-1646738917817)()]

NavigableMap让T有了对集合内元素搜索的能力

实现SortedMap接口让TreeMap有了对集合中的元素根据键排序的能力。默认key升序排序(也可以指定排序的比较器)

/**
 * @author shuang.kou
 * @createTime 2020年06月15日 17:02:00
 */
public class Person {
    private Integer age;

    public Person(Integer age) {
        this.age = age;
    }

    public Integer getAge() {
        return age;
    }


    public static void main(String[] args) {
        TreeMap<Person, String> treeMap = new TreeMap<>(new Comparator<Person>() {
            @Override
            public int compare(Person person1, Person person2) {
                int num = person1.getAge() - person2.getAge();
                return Integer.compare(num, 0);
            }
        });
        treeMap.put(new Person(3), "person1");
        treeMap.put(new Person(18), "person2");
        treeMap.put(new Person(35), "person3");
        treeMap.put(new Person(16), "person4");
        treeMap.entrySet().stream().forEach(personStringEntry -> {
            System.out.println(personStringEntry.getValue());
        });
    }
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33

综上,相比于HashMap来说 TreeMap 主要多了对集合中的元素根据键排序的能力以及对集合内元素的搜索的能力。

HashSet检查重复

​ 当你把对象加入HashSet时,HashSet 会先计算对象的hashcode值来判断对象加入的位置,同时也会与其他加入的对象的 hashcode 值作比较,如果没有相符的 hashcodeHashSet 会假设对象没有重复出现。但是如果发现有相同 hashcode 值的对象,这时会调用equals()方法来检查 hashcode 相等的对象是否真的相同。如果两者相同,HashSet 就不会让加入操作成功。

​ jdk8中,HashSet的add()方法只是简单调用了HashMap的put()方法,并且判断了一下返回值确保是否有重复元素。

// Returns: true if this set did not already contain the specified element
// 返回值:当set中没有包含add的元素时返回真
public boolean add(E e) {
        return map.put(e, PRESENT)==null;
}

//HashMap的putVal()方法中也有说明
// Returns : previous value, or null if none
// 返回值:如果插入位置没有元素返回null,否则返回上一个元素
final V putVal(int hash, K key, V value, boolean onlyIfAbsent,
                   boolean evict) {
...
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13

所以,无论HashSet中是否已经存在某元素,HashSet都会直接插入,只是会在add()方法返回值处告诉我们插入前是否存在相同元素。

HashMap底层实现

  • 结构:数组+链表,链表节点存储的是一个Entry对象,每个Entry对象存储四个属性(hash,key,value,next)
  • 初始化HashMap,有参和无参。无参中,初始容量为16,加载因子为0.75,阈值为16*0.75=12;

JDK1.7:

  1. 根据对冲突处理方式不同,哈希表两种实现方式:1、开放地址方式;2、冲突链表方式。java7采用冲突链表方式。

    有两个参数影响HashMap的性能:初始容量(inital capacity)负载系数(load factor)

    初始容量:指定初始table的大小

    负载系数:指定自动扩容的临界值

当entry数量超过capacity*load_factor时,容器自动扩容并重新哈希。对于插入元素较多的场景,将初始容量设大可以减少重新哈希次数。

将对象放入到HashMapHashSet中时,有两个方法需要特别关心: hashCode()equals()hashCode()方法决定了对象会被放到哪个bucket里,当多个对象的哈希值冲突时,equals()方法决定了这些对象是否是“同一个对象”。所以,如果要将自定义的对象放入到HashMapHashSet中,需要*@Override*hashCode()equals()方法。

  1. get()

    get(Object key)方法根据指定key返回对应的value,该方法调用了getEntry(Object key)得到相应的entry,然后返回entry.getValue()。因此,getEntry()是算法的核心算法思想是:首先通过Hash()函数得到对应bucket下标,然后依次遍历冲突链表,通过key.equals(k)方法来判断是否是要找的那个entry

HashMap_getEntry

上图中hash(k)&(table.length-1)等价于hash(k)%table.length,原因是HashMap要求table.length必须是2的指数,因此table.length-1就是二进制低位全是1,跟hash(k)相与会将哈希值的高位全抹掉,剩下的就是余数了。

//getEntry()方法
final Entry<K,V> getEntry(Object key) {
	......
	int hash = (key == null) ? 0 : hash(key);
    for (Entry<K,V> e = table[hash&(table.length-1)];//得到冲突链表
         e != null; e = e.next) {//依次遍历冲突链表中的每个entry
        Object k;
        //依据equals()方法判断是否相等
        if (e.hash == hash &&
            ((k = e.key) == key || (key != null && key.equals(k))))
            return e;
    }
    return null;
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  1. put()

    put(K key, V value)方法是将指定的key, value对添加到map里。该方法首先会对map做一次查找,看是否包含该元组,如果已经包含则直接返回,查找过程类似于getEntry()方法;如果没有找到,则会通过addEntry(int hash, K key, V value, int bucketIndex)方法插入新的entry,插入方式为头插法

    HashMap_addEntry
    • 先通过HM自己提供的hash算法算出当前key的hash值
    • 通过计算出的hash值调用indexFor方法计算当前对象应该存储在数组几号位置
    • 判断size是否达到阈值,没有继续,达到,数组扩容为两倍(size是当前容器中已有Entry的数量,不是数组长度。)
    • 将当前对应的hash,key,value封装成一个Entry,去数组中查找当前位置有无元素。没有放在这个位置;如果存在链表,遍历链表,若链表上某节点key与当前key进行equals后比较结果为true,将原来节点的value返回,将当前新的value替换为原来的value,若比较结果没有true,把刚才的Entry中next指向当前链表始节点,即第一个位置

    扩容机制:HM使用懒扩容,只会在put时才判断。注意,每次扩容后,都要重新计算原来的Entry在新数组中的位置。

    //addEntry()
    void addEntry(int hash, K key, V value, int bucketIndex) {
        if ((size >= threshold) && (null != table[bucketIndex])) {
            resize(2 * table.length);//自动扩容,并重新哈希
            hash = (null != key) ? hash(key) : 0;
            bucketIndex = hash & (table.length-1);//hash%table.length
        }
        //在冲突链表头部插入新的entry
        Entry<K,V> e = table[bucketIndex];
        table[bucketIndex] = new Entry<>(hash, key, value, e);
        size++;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
  2. remove()

//removeEntryForKey()
final Entry<K,V> removeEntryForKey(Object key) {
	......
	int hash = (key == null) ? 0 : hash(key);
    int i = indexFor(hash, table.length);//hash&(table.length-1)
    Entry<K,V> prev = table[i];//得到冲突链表
    Entry<K,V> e = prev;
    while (e != null) {//遍历冲突链表
        Entry<K,V> next = e.next;
        Object k;
        if (e.hash == hash &&
            ((k = e.key) == key || (key != null && key.equals(k)))) {//找到要删除的entry
            modCount++; size--;
            if (prev == e) table[i] = next;//删除的是冲突链表的第一个entry
            else prev.next = next;
            return e;
        }
        prev = e; e = next;
    }
    return e;
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21

JDK1.8

数组+ 链表+ 红黑树

java7中查找时根据hash值找到数组下标,但后面要顺着链表一个个比较下去才能找到,时间复杂度取决于链表长度,为O(n)。

为了降低这部分开销,java8中,当链表元素达到8个,会将链表转换为红黑树,在这里进行查找时间复杂度降为O(logN)。

java8使用Node代替entry,但只用于链表的情况,红黑树的情况使用TreeNode

  1. put()

    public V put(K key, V value) {
        return putVal(hash(key), key, value, false, true);
    }
    
    // 第四个参数 onlyIfAbsent 如果是 true,那么只有在不存在该 key 时才会进行 put 操作
    // 第五个参数 evict 我们这里不关心
    final V putVal(int hash, K key, V value, boolean onlyIfAbsent,
                   boolean evict) {
        Node<K,V>[] tab; Node<K,V> p; int n, i;
        // 第一次 put 值的时候,会触发下面的 resize(),类似 java7 的第一次 put 也要初始化数组长度
        // 第一次 resize 和后续的扩容有些不一样,因为这次是数组从 null 初始化到默认的 16 或自定义的初始容量
        if ((tab = table) == null || (n = tab.length) == 0)
            n = (tab = resize()).length;
        // 找到具体的数组下标,如果此位置没有值,那么直接初始化一下 Node 并放置在这个位置就可以了
        if ((p = tab[i = (n - 1) & hash]) == null)
            tab[i] = newNode(hash, key, value, null);
    
        else {// 数组该位置有数据
            Node<K,V> e; K k;
            // 首先,判断该位置的第一个数据和我们要插入的数据,key 是不是"相等",如果是,取出这个节点
            if (p.hash == hash &&
                ((k = p.key) == key || (key != null && key.equals(k))))
                e = p;
            // 如果该节点是代表红黑树的节点,调用红黑树的插值方法,本文不展开说红黑树
            else if (p instanceof TreeNode)
                e = ((TreeNode<K,V>)p).putTreeVal(this, tab, hash, key, value);
            else {
                // 到这里,说明数组该位置上是一个链表
                for (int binCount = 0; ; ++binCount) {
                    // 插入到链表的最后面(Java7 是插入到链表的最前面)
                    if ((e = p.next) == null) {
                        p.next = newNode(hash, key, value, null);
                        // TREEIFY_THRESHOLD 为 8,所以,如果新插入的值是链表中的第 8 个
                        // 会触发下面的 treeifyBin,也就是将链表转换为红黑树
                        if (binCount >= TREEIFY_THRESHOLD - 1) // -1 for 1st
                            treeifyBin(tab, hash);
                        break;
                    }
                    // 如果在该链表中找到了"相等"的 key(== 或 equals)
                    if (e.hash == hash &&
                        ((k = e.key) == key || (key != null && key.equals(k))))
                        // 此时 break,那么 e 为链表中[与要插入的新值的 key "相等"]的 node
                        break;
                    p = e;
                }
            }
            // e!=null 说明存在旧值的key与要插入的key"相等"
            // 对于我们分析的put操作,下面这个 if 其实就是进行 "值覆盖",然后返回旧值
            if (e != null) {
                V oldValue = e.value;
                if (!onlyIfAbsent || oldValue == null)
                    e.value = value;
                afterNodeAccess(e);
                return oldValue;
            }
        }
        ++modCount;
        // 如果 HashMap 由于新插入这个值导致 size 已经超过了阈值,需要进行扩容
        if (++size > threshold)
            resize();
        afterNodeInsertion(evict);
        return null;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63

相比起来,JDK1.7性能稍差,因为扰动了4次。

  • **拉链法:**将链表和数组相结合。

HashMap遍历方式

  1. 迭代器EntrySet

    public class HashMapTest {
        public static void main(String[] args) {
            // 创建并赋值 HashMap
            Map<Integer, String> map = new HashMap();
            map.put(1, "Java");
            map.put(2, "JDK");
            map.put(3, "Spring Framework");
            map.put(4, "MyBatis framework");
            map.put(5, "Java中文社群");
            // 遍历
            Iterator<Map.Entry<Integer, String>> iterator = map.entrySet().iterator();
            while (iterator.hasNext()) {
                Map.Entry<Integer, String> entry = iterator.next();
                System.out.println(entry.getKey());
                System.out.println(entry.getValue());
            }
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    1
    
    Java
    
    2
    
    JDK
    
    3
    
    Spring Framework
    
    4
    
    MyBatis framework
    
    5
    
    Java中文社群
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
  2. 迭代器keySet

    Iterator<Integer> iterator = map.keySet().iterator();
    while (iterator.hasNext()) {
        Integer key = iterator.next();
        System.out.println(key);
        System.out.println(map.get(key));
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    结果同上

  3. ForEach EntrySet

    for (Map.Entry<Integer, String> entry : map.entrySet()) {
        System.out.println(entry.getKey());
        System.out.println(entry.getValue());
    }
    
    • 1
    • 2
    • 3
    • 4

    同上

  4. ForEach KeySet

    for(Integer key : map.keySet()){
        System.out.println(key);
        System.out.println(map.get(key));
    }
    
    • 1
    • 2
    • 3
    • 4
  5. Lambda

    map.forEach((key, value)) -> {
        System.out.println(key);
        System.out.println(value);
    });
    
    • 1
    • 2
    • 3
    • 4
  6. Streams API 单线程

    map.entrySet().stream().forEach(entry) -> {
        System.out.println(entry.getKey());
        System.out.println(entry.getValue());
    });
    
    • 1
    • 2
    • 3
    • 4
  7. Streams API 多线程

    map.entrySet().parallelStream().forEach(entry) -> {
        //同上
    });
    
    • 1
    • 2
    • 3

    结果:

    4

    MyBatis framework

    5

    Java中文社群

    1

    Java

    2

    JDK

    3

    Spring Framework

  • 性能分析

使用迭代器或是forEach循环的EntrySet方法,性能是相同的。

使用…的keySet方法,性能也相同。

但是两种EntrySet比keySet方法性能高,是因为KeySet 在循环时使用了 map.get(key),而 map.get(key) 相当于又遍历了一遍 Map 集合去查询 key 所对应的值。(为什么要用“又”这个词?那是因为在使用迭代器或者 for 循环时,其实已经遍历了一遍 Map 集合了,因此再使用 map.get(key) 查询时,相当于遍历了两遍。)

EntrySet 只遍历了一遍 Map 集合,之后通过代码“Entry<Integer, String> entry = iterator.next()”把对象的 keyvalue 值都放入到了 Entry 对象中,因此再获取 keyvalue 值时就无需再遍历 Map 集合,只需要从 Entry 对象中取值就可以了。

所以,EntrySet 的性能比 KeySet 的性能高出了一倍,因为 KeySet 相当于循环了两遍 Map 集合,而 EntrySet 只循环了一遍

  • 安全性分析

我们不能在遍历中使用集合 map.remove() 来删除数据,这是非安全的操作方式,但我们可以使用迭代器的 iterator.remove() 的方法来删除数据,这是安全的删除集合的方式。同样的我们也可以使用 Lambda 中的 removeIf 来提前删除数据,或者是使用 Stream 中的 filter 过滤掉要删除的数据进行循环,这样都是安全的,当然我们也可以在 for 循环前删除数据在遍历也是线程安全的。

ConcurrentHashMap和HashTable

  • 底层数据结构:1.7采用分段的数组+链表,1.8采用和HashMap一样。

  • 实现线程安全的方式(※):①在 JDK1.7 的时候,ConcurrentHashMap(分段锁) 对整个桶数组进行了分割分段(Segment),每一把锁只锁容器其中一部分数据,多线程访问容器里不同数据段的数据,就不会存在锁竞争,提高并发访问率。 到了 JDK1.8 的时候已经摒弃了 Segment 的概念,而是直接用 Node 数组+链表+红黑树的数据结构来实现,并发控制使用 synchronized 和 CAS 来操作。(JDK1.6 以后 对 synchronized 锁做了很多优化) 整个看起来就像是优化过且线程安全的 HashMap,虽然在 JDK1.8 中还能看到 Segment 的数据结构,但是已经简化了属性,只是为了兼容旧版本。② Hashtable(同一把锁) :使用 synchronized 来保证线程安全,效率非常低下。当一个线程访问同步方法时,其他线程也访问同步方法,可能会进入阻塞或轮询状态,如使用 put 添加元素,另一个线程不能使用 put 添加元素,也不能使用 get,竞争会越来越激烈效率越低。

    JDK1.7的ConcurrentHashMap

段(Segment),每一把锁只锁容器其中一部分数据,多线程访问容器里不同数据段的数据,就不会存在锁竞争,提高并发访问率。 到了 JDK1.8 的时候已经摒弃了 Segment 的概念,而是直接用 Node 数组+链表+红黑树的数据结构来实现,并发控制使用 synchronized 和 CAS 来操作。(JDK1.6 以后 对 synchronized 锁做了很多优化) 整个看起来就像是优化过且线程安全的 HashMap,虽然在 JDK1.8 中还能看到 Segment 的数据结构,但是已经简化了属性,只是为了兼容旧版本。② Hashtable(同一把锁) :使用 synchronized 来保证线程安全,效率非常低下。当一个线程访问同步方法时,其他线程也访问同步方法,可能会进入阻塞或轮询状态,如使用 put 添加元素,另一个线程不能使用 put 添加元素,也不能使用 get,竞争会越来越激烈效率越低。

JDK1.7的ConcurrentHashMap
声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/羊村懒王/article/detail/733592
推荐阅读
相关标签
  

闽ICP备14008679号