ConcurrentHashMap解析

注:此文基于Android的ConcurrentHashMap
讲解ConcurrentHashMap之前,我先花点篇幅讲解下CAS的机制思想,因为ConcurrentHashMap的核心使用CAS。
CAS: (全称叫Compare And Swap)翻译过来是比较并交换。什么意思呢?其实它的思想就是在写时,先比较内存值(V)和旧的预期值(A)是否相等,如果相等则修改内存值为新值(B)并返回true;如果不相等则什么也不做返回false。CAS是可以确保原子操作的。
java中CAS通过UnSafe类实现,UnSafe类提供了硬件级别的原子操作,我们看下UnSafe源码:

public final class Unsafe {
    //UnSafe单例实例
    private static final Unsafe theUnsafe;
    //无效字段的偏移地址
    public static final int INVALID_FIELD_OFFSET = -1;
    /**
     ARRAY_XX_BASE_OFFSET代表某一类型的数组第一个元素的偏移地址
     ARRAY_XX_INDEX_SCALE 代表某一类型的数组中元素的增量地址,比如int此值为4
    */
    public static final int ARRAY_BOOLEAN_BASE_OFFSET;
    public static final int ARRAY_BYTE_BASE_OFFSET;
    public static final int ARRAY_SHORT_BASE_OFFSET;
    public static final int ARRAY_CHAR_BASE_OFFSET;
    public static final int ARRAY_INT_BASE_OFFSET;
    public static final int ARRAY_LONG_BASE_OFFSET;
    public static final int ARRAY_FLOAT_BASE_OFFSET;
    public static final int ARRAY_DOUBLE_BASE_OFFSET;
    public static final int ARRAY_OBJECT_BASE_OFFSET;
    public static final int ARRAY_BOOLEAN_INDEX_SCALE;
    public static final int ARRAY_BYTE_INDEX_SCALE;
    public static final int ARRAY_SHORT_INDEX_SCALE;
    public static final int ARRAY_CHAR_INDEX_SCALE;
    public static final int ARRAY_INT_INDEX_SCALE;
    public static final int ARRAY_LONG_INDEX_SCALE;
    public static final int ARRAY_FLOAT_INDEX_SCALE;
    public static final int ARRAY_DOUBLE_INDEX_SCALE;
    public static final int ARRAY_OBJECT_INDEX_SCALE;
    public static final int ADDRESS_SIZE;
    
    /**
        注册本地方法
    */
    private static native void registerNatives();

    private Unsafe() {
    }
    
    //获取UnSafe单例实例对象
    @CallerSensitive
    public static Unsafe getUnsafe() {
        Class var0 = Reflection.getCallerClass();
        if (!VM.isSystemDomainLoader(var0.getClassLoader())) {
            throw new SecurityException("Unsafe");
        } else {
            return theUnsafe;
        }
    }
    
    /**
    获取var1对象中偏移地址为var2的int类型的字段值
    var1:包含需要读取字段值的对象
    var2:  var1中int类型字段的偏移地址
    */
    public native int getInt(Object var1, long var2);
    /**
     设置var1对象中偏移地址为var2的int类型的字段值为var4
    */
    public native void putInt(Object var1, long var2, int var4);
    /**
    获取var1对象中偏移地址为var2、类型为Object的字段值
    */
    public native Object getObject(Object var1, long var2);
    /**
      设置var1对象中偏移地址为var2、类型为Object的字段值为var4
    */
    public native void putObject(Object var1, long var2, Object var4);
    /**
    获取var1对象中偏移地址为var2、类型为boolean的字段值
    */
    public native boolean getBoolean(Object var1, long var2);
    /**
    设置var1对象中偏移地址为var2、类型为boolean的字段值为var4
    */
    public native void putBoolean(Object var1, long var2, boolean var4);
    /**
    获取var1对象中偏移地址为var2、类型为byte的字段值
    */
    public native byte getByte(Object var1, long var2);
    /**
    设置var1对象中偏移地址为var2、类型为byte的字段值为var4
    */
    public native void putByte(Object var1, long var2, byte var4);
    /**
    获取var1对象中偏移地址为var2、类型为short的字段值
    */
    public native short getShort(Object var1, long var2);
    /**
    设置var1对象中偏移地址为var2、类型为short的字段值为var4
    */
    public native void putShort(Object var1, long var2, short var4);
    /**
    获取var1对象中偏移地址为var2、类型为char的字段值
    */
    public native char getChar(Object var1, long var2);
    /**
    设置var1对象中偏移地址为var2、类型为char的字段值为var4
    */
    public native void putChar(Object var1, long var2, char var4);
    /**
    获取var1对象中偏移地址为var2、类型为long的字段值
    */
    public native long getLong(Object var1, long var2);
    /**
    设置var1对象中偏移地址为var2、类型为long的字段值为var4
    */
    public native void putLong(Object var1, long var2, long var4);
    /**
    获取var1对象中偏移地址为var2、类型为float的字段值
    */
    public native float getFloat(Object var1, long var2);
    /**
    设置var1对象中偏移地址为var2、类型为float的字段值为var4
    */
    public native void putFloat(Object var1, long var2, float var4);
    /**
    获取var1对象中偏移地址为var2、类型为double的字段值
    */
    public native double getDouble(Object var1, long var2);
    /**
    设置var1对象中偏移地址为var2、类型为double的字段值为var4
    */
    public native void putDouble(Object var1, long var2, double var4);

     
    /**
    获取偏移地址为var1、类型为byte的字段值
    */
    public native byte getByte(long var1);
    /**
    设置偏移地址为var1、类型为byte的字段值为var3
    */
    public native void putByte(long var1, byte var3);
    /**
    获取偏移地址为var1、类型为short的字段值
    */
    public native short getShort(long var1);
    /**
    设置偏移地址为var1、类型为short的字段值为var3
    */
    public native void putShort(long var1, short var3);
    /**
    获取偏移地址为var1、类型为char的字段值
    */
    public native char getChar(long var1);
    /**
    设置偏移地址为var1、类型为char的字段值为var3
    */
    public native void putChar(long var1, char var3);
    /**
    获取偏移地址为var1、类型为int的字段值
    */
    public native int getInt(long var1);
    /**
    设置偏移地址为var1、类型为int的字段值为var3
    */
    public native void putInt(long var1, int var3);
    /**
    获取偏移地址为var1、类型为long的字段值
    */
    public native long getLong(long var1);
    /**
    设置偏移地址为var1、类型为long的字段值为var3
    */
    public native void putLong(long var1, long var3);
    /**
    获取偏移地址为var1、类型为float的字段值
    */
    public native float getFloat(long var1);
    /**
    设置偏移地址为var1、类型为float的字段值为var3
    */
    public native void putFloat(long var1, float var3);
    /**
    获取偏移地址为var1、类型为double的字段值
    */
    public native double getDouble(long var1);
    /**
    设置偏移地址为var1、类型为double的字段值为var3
    */
    public native void putDouble(long var1, double var3);
    /**
    获取偏移地址为var1、类型为long的字段值
    */
    public native long getAddress(long var1);
    /**
    获取偏移地址为var1、类型为long的字段值
    */
    public native void putAddress(long var1, long var3);
    /**
    请求分配内存var1大的内存,返回分配的起始地址
    */
    public native long allocateMemory(long var1);
    /**
    对偏移地址为var1的内存扩展var3的大小 
    */
    public native long reallocateMemory(long var1, long var3);
    /**
    
    */
    public native void setMemory(Object var1, long var2, long var4, byte var6);

    public void setMemory(long var1, long var3, byte var5) {
        this.setMemory((Object)null, var1, var3, var5);
    }

    public native void copyMemory(Object var1, long var2, Object var4, long var5, long var7);

    public void copyMemory(long var1, long var3, long var5) {
        this.copyMemory((Object)null, var1, (Object)null, var3, var5);
    }
    /**
    释放内存
    */
    public native void freeMemory(long var1);
    /**
    获取静态域中给定字段var1的内存偏移地址
    */
    public native long staticFieldOffset(Field var1);
    /**
    返回对象中给定静态字段的内存地址的偏移量,在这个类的其他方法中这个值只是被用作访问特定字段的方式。这个值对给定的字段是唯一的,并且后续对该方法的调用返回值都是一样的
    */
    public native long objectFieldOffset(Field var1);
    /**
    返货静态域的起始内存地址
    */
    public native Object staticFieldBase(Field var1);
    /**
    判断给定的类var1是否已经初始化,如果已经初始化则放回true,否则返回false
    */
    public native boolean shouldBeInitialized(Class<?> var1);
    /**
    确定给定的类var1初始化,如果没有初始化则进行初始化,否则不做任何处理
    */
    public native void ensureClassInitialized(Class<?> var1);
    /**
    返回给定类型为var1数组的起始地址
    */
    public native int arrayBaseOffset(Class<?> var1);
    /**
    返回给定类型为var1数组的内存地址增量
    */
    public native int arrayIndexScale(Class<?> var1);
    /**
    返回内存地址大小?不知道什么用
    */
    public native int addressSize();
    /**
    返回内存分页大小,不知道什么用
    */
    public native int pageSize();
    
    public native Class<?> defineClass(String var1, byte[] var2, int var3, int var4, ClassLoader var5, ProtectionDomain var6);

    public native Class<?> defineAnonymousClass(Class<?> var1, byte[] var2, Object[] var3);
    
    public native Object allocateInstance(Class<?> var1) throws InstantiationException;

    public native void throwException(Throwable var1);
    
    /**
    CAS操作,如果对象var1中编译地址为var2的对象(对应的是内存值)与预期值的对象var4相等,则更改内存值为对象var5,并返回true;封装什么也不做并返回false
    */
    public final native boolean compareAndSwapObject(Object var1, long var2, Object var4, Object var5);
    /**
    CAS操作,如果对象var1中编译地址为var2的int值(对应的是内存值)与预期值的int值var4相等,则更改内存值为int值var5,并返回true;封装什么也不做并返回false
    */
    public final native boolean compareAndSwapInt(Object var1, long var2, int var4, int var5);

    public final native boolean compareAndSwapLong(Object var1, long var2, long var4, long var6);
    /**
    获取对象var1中偏移地址为var2、类型为Object的字段值。此操作可以保证返回的值是内存中最新的值——可见性,即Volatile
    */
    public native Object getObjectVolatile(Object var1, long var2);
    /**
    设置对象var1中偏移地址为var2、类型为Object的字段值为var4。此操作可以保证设置内存值之后值可见性,即Volatile
    */
    public native void putObjectVolatile(Object var1, long var2, Object var4);
    /**
    获取对象var1中偏移地址为var2、类型为int的字段值。此操作可以保证返回的值是内存中最新的值——可见性,即Volatile
    */
    public native int getIntVolatile(Object var1, long var2);
    /**
    设置对象var1中偏移地址为var2、类型为int的字段值为var4。此操作可以保证设置内存值之后值可见性,即Volatile
    */
    public native void putIntVolatile(Object var1, long var2, int var4);
    /**
    获取对象var1中偏移地址为var2、类型为boolean的字段值。此操作可以保证返回的值是内存中最新的值——可见性,即Volatile
    */
    public native boolean getBooleanVolatile(Object var1, long var2);
    /**
    设置对象var1中偏移地址为var2、类型为boolean的字段值为var4。此操作可以保证设置内存值之后值可见性,即Volatile
    */
    public native void putBooleanVolatile(Object var1, long var2, boolean var4);

    //其他getXXVolatile、setXXVolatile一样的意思,不一一说明
    
    /**
    释放被var1创建的一个在线程上的阻塞。这个方法可以被使用来终止之前调用var1导致的阻塞。
    这个方法不是线程安全的,同时这个是java代码而不是native代码
    */
    public native void unpark(Object var1);
    /**
    阻塞一个线程知道unpark出现、线程被中断、超时。如果一个unpark调用已经出现了,
   * 这里只计数。timeout为0表示永不过期.当isAbsolute为true时,
   * timeout是相对于新纪元之后的毫秒。否则这个值就是超时前的纳秒数。这个方法执行时
   * 也可能不合理地返回(没有具体原因)
    */
    public native void park(boolean isAbsolute, long time);

    public native int getLoadAverage(double[] var1, int var2);
    
    /**
    获取对象var1中偏移地址为var2的值,如果内存中的int值与期望的值var5相等,则设置内存中的
值为var4+var5
    */
    public final int getAndAddInt(Object var1, long var2, int var4) {
        int var5;
        do {
            var5 = this.getIntVolatile(var1, var2);
        } while(!this.compareAndSwapInt(var1, var2, var5, var5 + var4));

        return var5;
    }

    public final long getAndAddLong(Object var1, long var2, long var4) {
        long var6;
        do {
            var6 = this.getLongVolatile(var1, var2);
        } while(!this.compareAndSwapLong(var1, var2, var6, var6 + var4));

        return var6;
    }

    public final int getAndSetInt(Object var1, long var2, int var4) {
        int var5;
        do {
            var5 = this.getIntVolatile(var1, var2);
        } while(!this.compareAndSwapInt(var1, var2, var5, var4));

        return var5;
    }

    public final long getAndSetLong(Object var1, long var2, long var4) {
        long var6;
        do {
            var6 = this.getLongVolatile(var1, var2);
        } while(!this.compareAndSwapLong(var1, var2, var6, var4));

        return var6;
    }

    public final Object getAndSetObject(Object var1, long var2, Object var4) {
        Object var5;
        do {
            var5 = this.getObjectVolatile(var1, var2);
        } while(!this.compareAndSwapObject(var1, var2, var5, var4));

        return var5;
    }

    public native void loadFence();

    public native void storeFence();

    public native void fullFence();
    
    private static void throwIllegalAccessError() {
        throw new IllegalAccessError();
    }

    static {
        //注册
        registerNatives();
        Reflection.registerMethodsToFilter(Unsafe.class, new String[]{"getUnsafe"});
        //实例化
        theUnsafe = new Unsafe();
        // 获取各种类型数组的起始偏移地址及其增量地址
        ARRAY_BOOLEAN_BASE_OFFSET = theUnsafe.arrayBaseOffset(boolean[].class);
        ARRAY_BYTE_BASE_OFFSET = theUnsafe.arrayBaseOffset(byte[].class);
        ARRAY_SHORT_BASE_OFFSET = theUnsafe.arrayBaseOffset(short[].class);
        ARRAY_CHAR_BASE_OFFSET = theUnsafe.arrayBaseOffset(char[].class);
        ARRAY_INT_BASE_OFFSET = theUnsafe.arrayBaseOffset(int[].class);
        ARRAY_LONG_BASE_OFFSET = theUnsafe.arrayBaseOffset(long[].class);
        ARRAY_FLOAT_BASE_OFFSET = theUnsafe.arrayBaseOffset(float[].class);
        ARRAY_DOUBLE_BASE_OFFSET = theUnsafe.arrayBaseOffset(double[].class);
        ARRAY_OBJECT_BASE_OFFSET = theUnsafe.arrayBaseOffset(Object[].class);
        ARRAY_BOOLEAN_INDEX_SCALE = theUnsafe.arrayIndexScale(boolean[].class);
        ARRAY_BYTE_INDEX_SCALE = theUnsafe.arrayIndexScale(byte[].class);
        ARRAY_SHORT_INDEX_SCALE = theUnsafe.arrayIndexScale(short[].class);
        ARRAY_CHAR_INDEX_SCALE = theUnsafe.arrayIndexScale(char[].class);
        ARRAY_INT_INDEX_SCALE = theUnsafe.arrayIndexScale(int[].class);
        ARRAY_LONG_INDEX_SCALE = theUnsafe.arrayIndexScale(long[].class);
        ARRAY_FLOAT_INDEX_SCALE = theUnsafe.arrayIndexScale(float[].class);
        ARRAY_DOUBLE_INDEX_SCALE = theUnsafe.arrayIndexScale(double[].class);
        ARRAY_OBJECT_INDEX_SCALE = theUnsafe.arrayIndexScale(Object[].class);
        ADDRESS_SIZE = theUnsafe.addressSize();
    }
}

总的来说,UnSafe类提供的功能主要有:
1、内存管理:

  • 分配内存:allocateMemory
  • 扩展内存:reallocateMemory
  • 释放内存:freeMemory
    2、读写对象字段值
  • 读:getXXX、objectFieldOffset、getXXXVolatile等
  • 写:putXXX、putXXXVolatile等
    3、线程挂起和恢复操作
  • 挂起:park
  • 恢复:unpark
    4、CAS操作: compareAndSwapInt、compareAndSwapObject、compareAndSwapLong等

我们了解了CAS和UnSafe的原理之后,接着我们开始解析ConcurrentHashMap:
1、ConcurrentHashMap的数据结构是什么?
2、ConcurrentHashMap默认容量是多少?
3、ConcurrentHashMap最大容量是多少?
4、ConcurrentHashMap每次扩容是多少?
5、ConcurrentHashMap散列函数的实现是?如何计算出散列地址?
6、ConcurrentHashMap get的原理是?
7、ConcurrentHashMap put的原理是?
8、ConcurrentHashMap 是否是线程安全?
9、多线程环境下为什么ConcurrentHashMap比HashTable性能好?
10、ConcurrentHashMap与HashMap的区别是?

问题1:ConcurrentHashMap的数据结构是什么?

ConcurrentHashMap也是采用数组+单链表的数据结构,其使用的是内部类中的Node作为节点

/**
     * The array of bins. Lazily initialized upon first insertion.
     * Size is always a power of two. Accessed directly by iterators.
     */
    transient volatile Node<K,V>[] table;

static class Node<K,V> implements Map.Entry<K,V> {
        final int hash;
        final K key;
        volatile V val;
        volatile Node<K,V> next;
        //...省略
}

通过上述代码,可以知道每个节点中hash值、key值是不可变的,而val和next是可见性的,即读写时均使用的内存中最新的值。

问题2:ConcurrentHashMap默认容量是多少?

与HashMap一样,其默认容量是16,看如下代码:

/**
     * The default initial table capacity.  Must be a power of 2
     * (i.e., at least 1) and at most MAXIMUM_CAPACITY.
     */
    private static final int DEFAULT_CAPACITY = 16;
问题3:ConcurrentHashMap最大容量是多少?

最大是其实是可以达到Integer.MAX_VALUE的,即使源码中定义了ConcurrentHashMap允许的最大容量,但是其在扩容的时候是不受这个最大容量的限制的,我们看下相关的源码:

private static final int MAXIMUM_CAPACITY = 1 << 30;

private final void transfer(Node<K,V>[] tab, Node<K,V>[] nextTab) {
        //n是当前散列表的大小
        int n = tab.length, stride;
                   //...省略
        //nextTab用于扩容时,存放移动或者拷贝原散列表元素,结束之后将nextTab赋值给table的
        if (nextTab == null) {            // initiating
            try {
                @SuppressWarnings("unchecked")
                /**扩容大小每次为n*2,并没有使用最大值进行限制,所以时可以达到
                Integer.MAX_VALUE的
                */
                Node<K,V>[] nt = (Node<K,V>[])new Node<?,?>[n << 1];
                nextTab = nt;
            } catch (Throwable ex) {      // try to cope with OOME
                sizeCtl = Integer.MAX_VALUE;
                return;
            }
            //...省略
      }
            //...省略
}

上述代码是在扩容时,默认扩容的大小为原来容量的2倍,且不受定义中最大容量的限制,所以其最大容量其实是可以达到Integer.MAX_VALUE的

问题4:ConcurrentHashMap每次扩容是多少?

每次扩容时原来的容量的2倍,即:n << 1。同时android中的ConcurrentHashMap与HashMap不一样有个扩容阈值和加载因子。

问题5:ConcurrentHashMap散列函数的实现是?如何计算出散列地址?

此问题在问题6中回答

问题6:ConcurrentHashMap get的原理是?

我们先看get的源码部分:

public V get(Object key) {
        /**
        tab:散列表
        e、p: 散列表中的节点
        n:散列表容量大小
        eh: 匹配节点的hash值
        ek: 匹配节点的key值
        h: 要查找的hash值
        */
        Node<K,V>[] tab; Node<K,V> e, p; int n, eh; K ek;
        //通过spread函数来对key的hashcode继续计算,得到key的hash值
        int h = spread(key.hashCode());
        // 通过(n -1)& h得到映射地址即对于的散列表下标
        if ((tab = table) != null && (n = tab.length) > 0 &&
            (e = tabAt(tab, (n - 1) & h)) != null) {
            //如果散列表不为空,并且取下标处的节点不为空
            /**
            如果e代表的链表表头节点的hash值与要查找的has值相等,并且key相等,则代表表头节点就是查找
            的,直接返回此节点val值
            */
            if ((eh = e.hash) == h) {
                if ((ek = e.key) == key || (ek != null && key.equals(ek)))
                    return e.val;
            }
            else if (eh < 0)
                // 如果hash值小0,在以表头为e的链表中查找匹配的节点并返回节点的值
                return (p = e.find(h, key)) != null ? p.val : null;
            while ((e = e.next) != null) {
                //遍历链表,知道找到匹配的节点,并返回节点的值
                if (e.hash == h &&
                    ((ek = e.key) == key || (ek != null && key.equals(ek))))
                    return e.val;
            }
        }
        return null;
    }
// 对key的hashcode进行(h ^(h>>>16)) & HASH_BITS操作,得到key的hash值
static final int spread(int h) {
        return (h ^ (h >>> 16)) & HASH_BITS;
    }
/**
  通过UnSafe对象的getObjectVolatile方法去除数组tab中下标为((long)i << ASHIFT) + ABASE的节点,其
  具有Volatile语义,即同步内存的值,取到的节点是内存中最新的值
*/
static final <K,V> Node<K,V> tabAt(Node<K,V>[] tab, int i) {
        return (Node<K,V>)U.getObjectVolatile(tab, ((long)i << ASHIFT) + ABASE);
    }

get的原理总结来说是:
1、先计算出key的hash值:取key的hashcode进行(h ^ (h>>>16)) & HASH_BITS得到key的hash值
2、计算出映射地址及散列表的下标:对key的hash值进行h & (n - 1)操作得到映射地址
3、根据下标取出链表,再在链表中查找匹配的节点,如果找到节点则返回节点的val;如果找不到节点则返回null
到此我们可以很容易回答问题5中的问题:
散列函数的实现是:取key的hashcode,对hashcode进行(h ^ (h >>> 16)) & HASH_BITS操作得到hash值
是这样计算出映射地址的:对key的hash值进行h & (n - 1) 操作得到映射地址

问题7:ConcurrentHashMap put的原理是?

我们先看下ConcurrentHashMap中put的源码:

public V put(K key, V value) {
        return putVal(key, value, false);
    }
final V putVal(K key, V value, boolean onlyIfAbsent) {
        // 如果key或者value为null则抛出NullPointerException
        if (key == null || value == null) throw new NullPointerException();
        //计算出key的hash值
        int hash = spread(key.hashCode());
        //链表大小
        int binCount = 0;
        // 遍历散列表
        for (Node<K,V>[] tab = table;;) {
            /**
            f: 散列表中的节点
            n: 散列表数组的长度
            i: 散列表的下标
            fh: 节点的hash值
            */
            Node<K,V> f; int n, i, fh;

            if (tab == null || (n = tab.length) == 0)
                // 如果散列表为空,则初始化散列表
                tab = initTable();
            else if ((f = tabAt(tab, i = (n - 1) & hash)) == null) {
                /** 
                如果散列表中下标为i的节点为空,即散列表中不存在此键值对,则先创建节点,通过cas操作
                将节点存入散列表的i位置,并退出循环
                */
                if (casTabAt(tab, i, null,
                             new Node<K,V>(hash, key, value, null)))
                    break;                   // no lock when adding to empty bin
            }
            else if ((fh = f.hash) == MOVED)
                //如果散列表正在移动,及扩容操作,则使用helpTransfer方法协助其他线程一同扩容
                tab = helpTransfer(tab, f);
            else {
                // 节点的旧值
                V oldVal = null;
                synchronized (f) {
                    // 从新从内存中取散列表i位置的节点
                    if (tabAt(tab, i) == f) {
                        //如果内存中最新的节点与f相等,则进行下列操作
                        if (fh >= 0) {
                            //如果hash值大于0,为什么?
                            binCount = 1;
                            for (Node<K,V> e = f;; ++binCount) {
                                /**
                                  遍历链表,判断链表中是否已经存在要传入的元素,如果已经存在则
                                  替换找到的元素的值,并返回旧值
                                */
                                K ek;
                                if (e.hash == hash &&
                                    ((ek = e.key) == key ||
                                     (ek != null && key.equals(ek)))) {
                                    oldVal = e.val;
                                    if (!onlyIfAbsent)
                                        e.val = value;
                                    break;
                                }
                                Node<K,V> pred = e;
                                if ((e = e.next) == null) {
                                    /**
                                    已经遍历到链表的表尾还没找到要添加的元素,则代表此元素在链表中时不存在,是
                                    新的元素,则创建新节点并加入链表的表尾
                                    */
                                    pred.next = new Node<K,V>(hash, key,
                                                              value, null);
                                    break;
                                }
                            }
                        }
                        else if (f instanceof TreeBin) {
                            // 如果节点是树形结构,则通过树的方式进行put操作,在此不做说明
                            Node<K,V> p;
                            binCount = 2;
                            if ((p = ((TreeBin<K,V>)f).putTreeVal(hash, key,
                                                           value)) != null) {
                                oldVal = p.val;
                                if (!onlyIfAbsent)
                                    p.val = value;
                            }
                        }
                        else if (f instanceof ReservationNode)
                            throw new IllegalStateException("Recursive update");
                    }
                }
                if (binCount != 0) {
                    if (binCount >= TREEIFY_THRESHOLD)
                        treeifyBin(tab, i);
                    if (oldVal != null)
                        return oldVal;
                    break;
                }
            }
        }
        // 进行扩容处理
        addCount(1L, binCount);
        return null;
    }

put的原理总结如下:
1、计算出key的hash值
2、通过key的hash值计算出其对应的映射地址及散列表的下标
3、取出散列表下标处的链表,如果链表为空,代表是新的元素,通过cas进行添加操作
4、如果散列表正在扩容,则协助其他线程一起完成扩容操作,再重新进行开始步骤3的操作
5、接着通过遍历链表,判断是否已经存在要插入的元素,如果存在则更改此元素的值并返回旧的值
6、如果在链表中没有找到要插入的元素,则通过cas进行添加操作
7、如果是添加元素则还需要根据需要进行扩容操作

问题8:ConcurrentHashMap 是否是线程安全?

答案是肯定的,ConcurrentHashMap是线程安全的。通过上述代码可以看到ConcurrentHashMap使用了Volatile、cas、synchronized的语义,采用锁分离机制确保线程的安全。其中get的操作时通过volatile来确保读取的是内存中最新的值;put操作通过volatile找出对应的链表,通过cas的方式创建链表;通过synchronized遍历链表并进行元素值的更改或者添加元素操作。

问题9:多线程环境下为什么ConcurrentHashMap比HashTable性能好?

问题8中的分析,而HashTable加锁机制是在每个涉及读写操作的方法都使用synchronized修饰。所以ConcurrentHashMap的get操作是具有并发性的,put操作则具有局部并发功能,所以ConcurrentHashMap的性能远比HashTable好。

问题10:ConcurrentHashMap与HashMap的区别是?

主要区别是:
1、ConcurrentHashMap是线程安全类,而HashMap不是。所以在单线程中ConcurrentHashMap性能比HashMap的性能差,但ConcurrentHashMap在多线程中时安全的而HashMap在多线程是不是安全的
2、ConcurrentHashMah不允许key和value为null,而HashMap是允许的
3、ConcurrentHashMap与HashMap默认容量都是16
4、ConcurrentHashMap与HashMap每次扩容默认都是原来容量的2倍

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,001评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,210评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,874评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,001评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,022评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,005评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,929评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,742评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,193评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,427评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,583评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,305评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,911评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,564评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,731评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,581评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,478评论 2 352

推荐阅读更多精彩内容