JUC并发编程之HashMap(jdk1.7版本)-底层源码探究

本文详细探讨了JDK1.7中HashMap的底层实现,包括数据结构(数组+链表),哈希冲突处理,头插法可能导致的链表环问题,以及扩容机制。文章还分析了为何HashMap的容量必须为2的幂次以及负载因子设为0.75的原因,并介绍了多线程环境下扩容可能导致的死锁问题。在JDK1.8中,这些问题得到了优化,采用了红黑树和不同的扩容策略以避免死锁。
摘要由CSDN通过智能技术生成

目录

JUC并发编程之HashMap(jdk1.7版本)-底层源码探究

HashMap底层源码 - jdk1.7

基本概念 -采取层层递进,问答式

存储Key-Value的结构

常量和成员变量

构造方法

put方法

inflateTable方法

hash方法

indexFor方法

addEntry方法

resize方法

createEntry方法

get相关方法

size方法

remove相关方法

jdk1.7版本的HashMap为什么会产生死锁(链表成环)?

出现问题的场景:

解决方法:


JUC并发编程之HashMap(jdk1.7版本)-底层源码探究

HashMap在jdk1.7和jdk1.8的区别非常大,jdk1.8中引入了红黑树来优化性能,并且jdk1.8解决了jdk1.7版本下多线程并发时的死锁问题,为了层层递进学习,先从jdk1.7的HashMap底层开始学起

首先要学会如何更改所处模块的jdk版本号:

HashMap底层源码 - jdk1.7

基本概念 -采取层层递进,问答式

熟悉以下概念后,更容易进行阅读源码!!!

(1)jdk1.7版本的HashMap底层的数据结构为:数组+链表

(2) HashMap集合的大小容量就是等于数组的大小

(3) new HashMap()倘若不指明初始化大小,那么默认设置为16

(4) 但即使指定了HashMap的初始化容量,若该容量不是2的次方,那么会找一个最接近该指定初始化大小的数大于该指定初始大小的数是2的次方的数作为HashMap的初始化容量

eg:new HashMap(9) ->最终HashMap生成时初始化大小为16

(5) HashMap哈希表put插入元素时,如果发生哈希碰撞,那么对比发生哈希碰撞的两个Entry的键值是否一致,若一致,覆盖前一个插入哈希表的Entry的value值。若不一致,那么采用头插法插入到前一个Entry的头部,经过一次遍历后,插入头部的Entry会下移变为第一个元素节点。

(6) HashMap的get,put操作的时间复杂度为O(1)。其实还分最好和最坏的情况。

get:

最好情况:无哈希碰撞,所有Entry都插入到数组的各个位置,查询元素节点时时间复杂度为O(1),因为数组底层维护的有一个索引,可以直接定位到任意索引位置处的元素。

最坏情况:发生哈希碰撞,并且所要get的Entry正好插在链表的尾部。那么最坏时间复杂度为O(n),n为链表的长度。【注释:但是这种情况的概率是极小的,因为哈希碰撞频率实际上不是十分多,即使存在,也比较少出现节点是在尾部的这种极端情况,因为jdk1.7版本HashMap采取头插法就注定寻找节点Entry处于尾部的概率是极低的

平均时间复杂度:O(1)

put:

最好情况:无哈希碰撞,O(1)

最坏情况:有哈希碰撞,但是jdk1.7采取链地址法和头插法,所以无需遍历再插入,而是直接插在头部,所以时间复杂度最坏情况下也是O(1)

平均时间复杂度:O(1)

 (7) 哈希表底层数组的所有的元素位是否能够被100%利用起来?

no。虽然哈希冲突的概率是比较低的,但是依旧存在,所以存在哈希冲突碰撞。所以引入链表数据结构使用链地址法进行解决hash冲突,并且采用头插法插入Entry节点到链表中。

(8) 位运算(&,|,^,~)的计算效率是远大于+,-,*,%的。 位运算最接近机器语言,即是最接近底层语言,因为CPU底层只认二进制的机器语言!所以位运算效率最高。+ - * %中 % 的运算效率最低。

(9) 前面记录了,new HashMap(初始容量大小size),初始化时如果指定的容量非2的次方大小时,底层强制把该初始容量大小转化为2的指数次幂

怎么转化?必须满足以下三个条件:

1. 必须最接近size

2.必须>=size

3.必须是2的次方幂

eg:new HashMap(17),size=17,强制转化为指定初始化容量为:32

(10) 为什么HashMap初始化容量一定要转化为2的指数次幂??【重点】

前面提过,HashMap的容量即是底层数组的大小。每一个Entry节点插入到哈希表中,都需要先获取到一个索引,根据这个索引再确定是插入到HashMap底层的数组上的哪一个位置。

如何获取该索引的呢?底层源码如下:

计算索引:int i = indexFor(hash, table.length);
static int indexFor(int h, int length) {
//  key.hashCode % table.lenth
	return h & (table.lenth-1);
}

采取的是h&(数组长度-1),h代表的是Entry节点对应键值key的hashCode值,每一个类都继承Object类,Object类中有一个方法:hashCode()。通过key.hashCode()就可以获取得到一个哈希值

该哈希值可能是一个非常大的数字:137812378917823.....

如何通过该数字得到该哈希值对应的Entry应该插入到数组的哪一个索引位置呢?

最初思考到的肯定是:key.hashCode() % (数组长度-1),得到的范围为:[0,数组长度-1),这正好对应的是数组的所有索引下标对应的范围!

但是这种 % 的方式是十分低性能的!

通过测试对比出,位运算的性能高于 % 运算的十倍左右 !

所以我们想要使用 位运算进行优化!!!但是位运算结合哈希值(随机大的数值)来求出一个数组索引的范围怎么去求??

数组的索引范围= key.hashCode() & (数组长度-1)注释:数组长度必须为2的n次幂

自己举一个例子试一试就明白啦:如 数组长度为16

16 - 1 = 0000 0000 0001 0000 - 1 = 0000 0000 0000 1111

& 运算的性质为:两个都为1才为1,有一个为0就为0。所以2^n这个数字保证了2^n-1这个数值的二进制为有值的地方全为1。

(11) 何时扩容?谈谈扩容机制

threshold(扩容阙值) = capacity(数组的长度) * 0.75[扩容阙值比率,负载因子] = 16 * 0.75 = 12

当hashMap中存储的节点Entry元素的数量size,size>= threshold时,那么进行扩容!

扩容怎么扩?

(1) 大小扩容到原来的2倍。

(2) 转移数据,通过transfer方法

void transfer(Entry[] newTable, boolean rehash) {
        int newCapacity = newTable.length;
        for (Entry<K,V> e : table) {
            while(null != e) {
                Entry<K,V> next = e.next;
                if (rehash) { 
                    e.hash = null == e.key ? 0 : hash(e.key);//再一次进行hash计算?
                }
                int i = indexFor(e.hash, newCapacity);
                e.next = newTable[i];
                newTable[i] = e;
                e = next;
            }
        }
    }
    

多线程执行情况下,由于CPU线程随机调度可能会导致链表成环,最终导致死锁问题。jdk1.8的HashMap解决了该问题。之后会细谈该问题!

成环的原因:在多线程的情况下,并且可能原来发生哈希冲突而形成一条链表上的节点们,再一次由于哈希冲突而加到一个位置处,由于是头插法,所以位置会颠倒过来,再经过一次循环后,极易形成环。导致死环,死锁问题。

(12) hash扩容时,有一个加载因子,为什么该加载因子loadFactor为0.75 ?

其实0.75也不是最佳答案,通过牛顿二项式可求出,基于空间和时间的折中最佳加载因子为0.693

解析:首先要明白一点,当哈希表中的元素数量size >= threshold(数组容量*加载因子)时,我们需要进行2倍扩容。

所以加载因子loadFactor越大,那么threshold值就越大,threshold越大,那么扩容的概率就越小,那么哈希冲突的概率就越大。

但是加载因子过小的话,那么threshold值就越小,threshold越小,那么扩容的概率就越大,那么哈希冲突的概率就越小,虽然哈希冲突的概率变小了,但是浪费的数组空间大小变多了,一味的进行二倍扩容会导致数组越来越大,导致空间复杂度增加。

所以我们需要考虑一个空间和时间的折中的最佳加载因子,通过牛顿二项式计算出最佳值为0.693

存储Key-Value的结构

static class Entry<K,V> implements Map.Entry<K,V> {
        final K key;
        V value;
        Entry<K,V> next;
        int hash;

        /**
         * Creates new entry.
         */
        Entry(int h, K k, V v, Entry<K,V> n) {
            value = v;
            next = n;
            key = k;
            hash = h;
        }
}

常量和成员变量

	//默认初始化容量,16
    static final int DEFAULT_INITIAL_CAPACITY = 1 << 4; // aka 16

    //最大初始化容量,2^30
    static final int MAXIMUM_CAPACITY = 1 << 30;
	
	//默认负载因子,0.75
    static final float DEFAULT_LOAD_FACTOR = 0.75f;
	
	//默认初始化table的空数组
	//HashMap采用的是一种延迟加载的机制:当HashMap被创建时table被初始化为一个空数组,只有当其被使用
	//时,才创建一个非空数组。
    static final Entry<?,?>[] EMPTY_TABLE = {};

    /**
     * The table, resized as necessary. Length MUST Always be a power of two.
     * HashMap底层数组结构中的数组,在必要时可以进行扩容,但是数组的length必须为2的整数次幂
     *
     *
     * (Question1:为什么数组的长度必须是2的整数次幂?)
     */
    transient Entry<K,V>[] table = (Entry<K,V>[]) EMPTY_TABLE;

	//HashMap中存储的key-value键值对的数量,也就是存储的Entry节点的数量
    transient int size;
	
	//扩容阈值,同时也代表刚创建HashMap时的initialCapacity
    int threshold;

    //扩容因子,与HashMap扩容有关,threshold = loadFactor * capacity
    final float loadFactor;

    /**
     * The number of times this HashMap has been structurally modified
     * Structural modifications are those that change the number of mappings in
     * the HashMap or otherwise modify its internal structure (e.g.,
     * rehash).  This field is used to make iterators on Collection-views of
     * the HashMap fail-fast.  (See ConcurrentModificationException).
     */
	/**
	 * modCount用来记录HashMap发生结构性修改的次数,比如添加元素、删除元素等(在阅读后面的方法时我们会记		
	 * 录一下哪些情况算是结构性修改,Question2),该变量是用于HashMap集合视图中的迭代器的fail-fast策
	 * 略。
	 * 
	 * 在创建迭代器时,会将modCount赋值给一个名为expectedModCount的变量。在当前线程使用迭代器的过程
	 * 中,会不断地校验modCount与expectedModCount是否相等。如果二者值不相等,根据fail-fast策略,会
	 * 立即抛出ConcurrentModificationException,从而实现不让其他线程对HashMap进行结构性的修改。
	 * 可以参考内部类HashIterator的代码。
	 *
	 * fail-fast策略是一种错误检测策略,但无法避免错误。它是java集合中的一种错误机制,当多个线程同时对
	 * 一个集合进行修改时,就会发生ConcurrentModificationException。所以在并发环境下,还是建议使用
	 * j.u.c包下的组件。
	 */
    transient int modCount;

构造方法

	//无参构造方法
	public HashMap() {
        this(DEFAULT_INITIAL_CAPACITY, DEFAULT_LOAD_FACTOR);
    }
	//带一个参数的构造方法
	public HashMap(int initialCapacity) {
        this(initialCapacity, DEFAULT_LOAD_FACTOR);
    }
	//传入初始容量以及扩容因子的方法
	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;
        //刚创建HashMap时将初始化容量记录到threshold中
        threshold = initialCapacity;
        //空方法,LinkedHashMap中使用到
        init();
    }
	public HashMap(Map<? extends K, ? extends V> m) {
        this(Math.max((int) (m.size() / DEFAULT_LOAD_FACTOR) + 1,
                      DEFAULT_INITIAL_CAPACITY), DEFAULT_LOAD_FACTOR);
        inflateTable(threshold);

        putAllForCreate(m);
    }

put方法

put方法是用于向HashMap中添加Key-Value键值对的方法。若要添加的Key已存在于HashMap中,用传入的value值覆盖原来的oldValue,并将oldValue返回;若不存在,则直接添加,并返回null。

	/**
     * Associates the specified value with the specified key in this map.
     * If the map previously contained a mapping for the key, the old
     * value is replaced.
     */
	//向HashMap中添加元素的方法
	public V put(K key, V value) {
        //当第一次调用put方法时才对table进行初始化
        if (table == EMPTY_TABLE) {
            //创建table
            inflateTable(threshold);
        }
        //由此可见,jdk1.7版本下的HashMap支持Key为null的键值对
        //如果要put元素的key为null,则直接将该元素存储到table[0]链表中
        if (key == null)
            return putForNullKey(value);
        //根据key散列出hash值
        int hash = hash(key);
        //根据hash值和table的长度计算出该元素应插入的链表在table中的下标i
        int i = indexFor(hash, table.length);
        //在table[i]中寻找与插入元素key相同的元素
        for (Entry<K,V> e = table[i]; e != null; e = e.next) {
            Object k;
            /**
             * 注意此处HashMap是如何判断key相等的:
             * e.hash == hash && ((k = e.key) == key || key.equals(k))
             * 计算hash时也使用到了key的hashcode方法
             *
             * 所以,当key的类型是自定义类型,如果重写了equals方法,那么同时也要重写hashCode方法
             * 防止出现key相同,但是经过hashCode方法散列后的hash不同的情况
             */
            if (e.hash == hash && ((k = e.key) == key || key.equals(k))) {
                V oldValue = e.value;
                e.value = value;
                e.recordAccess(this);
                return oldValue;
            }
        }
	    //方法执行到此处时,说明原链表中不存在与插入元素key相同的元素,那么,就需要创建一个Entry并插入
        //向HashMap添加一个元素时,modCount需要自增
        modCount++;
        //添加Entry
        addEntry(hash, key, value, i);
        return null;
    }
	//putForNullKey方法是进行key为null的情况下的插入操作
	private V putForNullKey(V value) {
        //没有求hash,也没有求i,直接从table[0]中查找是否有Key相同的元素
        for (Entry<K,V> e = table[0]; e != null; e = e.next) {
            if (e.key == null) {
                V oldValue = e.value;
                e.value = value;
                e.recordAccess(this);
                return oldValue;
            }
        }
        modCount++;
        addEntry(0, null, value, 0);
        return null;
    }

inflateTable方法

inflateTable方法是根据创建HashMap时传入的初始容量或者默认初始容量来创建数组,并且初始化table数组。

	//方法参数toSize就是HashMap初始容量
	private void inflateTable(int toSize) {
        // roundUpToPowerOf2是根据初始容量计算出一个值capacity,作为table的长度
        // 该值满足:capacity >= toSize,并且capacity为2的整数次幂
        int capacity = roundUpToPowerOf2(toSize);
	    // 重新计算扩容阈值:threshold = capacity * loadFactor
        threshold = (int) Math.min(capacity * loadFactor, MAXIMUM_CAPACITY + 1);
        //创建数组
        table = new Entry[capacity];
        initHashSeedAsNeeded(capacity);
    }
	private static int roundUpToPowerOf2(int number) {
        // assert number >= 0 : "number must be non-negative";
        return number >= MAXIMUM_CAPACITY
                ? MAXIMUM_CAPACITY
                : (number > 1) ? Integer.highestOneBit((number - 1) << 1) : 1;
    }
	//该方法是求出i的最高位,比如9对应的2进制为:1001,经过该运算后,求出结果为1000
	public static int highestOneBit(int i) {
        //该方法是通过多次或运算,将i的低位全都变成1,最后再进行右移再相减,就只保留了最高位的1
        //如:1001,经过五次或运算,变成1111,最后一步为1111 - 0111 = 1000
        i |= (i >>  1);
        i |= (i >>  2);
        i |= (i >>  4);
        i |= (i >>  8);
        i |= (i >> 16);
        return i - (i >>> 1);
    }

hash方法

​hash方法是根据key计算出对应的hash值,这个hash值在定位插入链表在table中的下标(indexFor)时会使用到。

这个hash方法的作用在于:可以是得到的哈希值更加散列均匀,这样减少了哈希冲突,提升算法散列的性能。

	//HashMap中的hash算法要求算法散列性尽可能的高
	final int hash(Object k) {
        int h = hashSeed;
        if (0 != h && k instanceof String) {
            return sun.misc.Hashing.stringHash32((String) k);
        }

        h ^= k.hashCode();

        // 通过多次位运算,提高算法散列性
        h ^= (h >>> 20) ^ (h >>> 12);
        return h ^ (h >>> 7) ^ (h >>> 4);
    }

indexFor方法

​ indexFor方法就是根据hash值和table长度计算出插入链表在table中的下标i。

	static int indexFor(int h, int length) {
        /**
         * 计算下标i,可以使用取模%操作,也可以使用按位&操作,但是计算机底层运算实际上还是2进制的位运
         * 算,所以按位&操作效率会更高。
         * 
         *
         * 此处就可以解释Q1:为什么table的长度必须为2的整数次幂?
         * 因为我们此处求下标i使用的是按位&操作,如果length - 1中某一位为0,
         * 则该位上按位&操作必然为0,如:length为1011
         * length - 1:1010,
         * 则进行按位与操作时,数组上的有些位置将永远访问不到,造成空间的浪费,而且也增加了
         * hash冲突的可能性。而如果length满足2的整数次幂,那么put操作时要插入的元素可以被散列到数组的所
         * 有位置。
         */
        return h & (length-1);
    }

因为当length为2的整数次幂时,待插入元素散列到数组中任一位置的几率一样,也就是有机会可以被散列到table中的任一位置,可以有效利用数组空间,也可以减少hash冲突的可能性。

addEntry方法

​addEntry方法执行时,需要先判断数组是否需要扩容,再进行元素添加。

	void addEntry(int hash, K key, V value, int bucketIndex) {
        //jdk1.7版本HashMap的扩容条件:(size >= threshold) && (null != table[bucketIndex])
        //扩容条件:1、当前HashMap中Entry个数 >= threshold 2、要插入位置的链表不为空
        //jdk1.7和1.8中HashMap的扩容条件有一些差异,需要注意!!!
        if ((size >= threshold) && (null != table[bucketIndex])) {
            //扩容,新数组的长度为原数组的2倍
            resize(2 * table.length);
            hash = (null != key) ? hash(key) : 0;
            //扩容后需要重新计算index
            bucketIndex = indexFor(hash, table.length);
        }
        createEntry(hash, key, value, bucketIndex);
    }
	

 通过该方法,我们可以知道jdk1.7下HashMap扩容的两个条件,以及HashMap扩容后的数组长度为原数组的2倍。

resize方法

resize方法是对HashMap进行扩容,并将原table中的元素转移到新table中。多线程情况下进行扩容会导致出现循环死锁的情况。

	void resize(int newCapacity) {
        Entry[] oldTable = table;
        int oldCapacity = oldTable.length;
        if (oldCapacity == MAXIMUM_CAPACITY) {
            threshold = Integer.MAX_VALUE;
            return;
        }
		//创建新数组
        Entry[] newTable = new Entry[newCapacity];
        //将原table中的元素转移到新table中
        transfer(newTable, initHashSeedAsNeeded(newCapacity));
        table = newTable;
        //重新计算扩容阈值
        threshold = (int)Math.min(newCapacity * loadFactor, MAXIMUM_CAPACITY + 1);
    }
	//转移元素
	void transfer(Entry[] newTable, boolean rehash) {
        int newCapacity = newTable.length;
        for (Entry<K,V> e : table) {
            while(null != e) {
                Entry<K,V> next = e.next;
                if (rehash) {
                    e.hash = null == e.key ? 0 : hash(e.key);
                }
                int i = indexFor(e.hash, newCapacity);
                e.next = newTable[i];
                newTable[i] = e;
                e = next;
            }
        }
    }

createEntry方法

createEntry方法是真正进行创建Entry并且插入链表操作的方法。该方法中将新创建的Entry通过头插法插入到链表中。jdk1.7版本下HashMap插入时采取的是头插法。java开发者认为新插入的Entry可能会更多地被访问,所以为了方便以后的存取,将新添加的元素插入到链表头部。但是该插入策略会使得在 扩容后的transfer方法中可能会产生死循环链表,所以在jdk1.8开始就改成了尾插法。

	void createEntry(int hash, K key, V value, int bucketIndex) {
        Entry<K,V> e = table[bucketIndex];
        table[bucketIndex] = new Entry<>(hash, key, value, e);
        size++;
    }

get相关方法

​ get相关的方法在内部主要将具体的get操作委托给了getEntry方法。

	public V get(Object key) {
        if (key == null)
            return getForNullKey();
        Entry<K,V> entry = getEntry(key);

        return null == entry ? null : entry.getValue();
    }
	private V getForNullKey() {
        if (size == 0) {
            return null;
        }
        for (Entry<K,V> e = table[0]; e != null; e = e.next) {
            if (e.key == null)
                return e.value;
        }
        return null;
    }
	public boolean containsKey(Object key) {
        return getEntry(key) != null;
    }
	final Entry<K,V> getEntry(Object key) {
        if (size == 0) {
            return null;
        }

        int hash = (key == null) ? 0 : hash(key);
        for (Entry<K,V> e = table[indexFor(hash, table.length)];
             e != null;
             e = e.next) {
            Object k;
            if (e.hash == hash &&
                ((k = e.key) == key || (key != null && key.equals(k))))
                return e;
        }
        return null;
    }

size方法

​size方法返回HashMap中存储的Entry数量。看过该方法代码后可以方便区分字符串的length()方法、数组的length、集合的size()方法。

	public int size() {
        return size;
    }

remove相关方法

​ remove、clear相关操作可能会对HashMap产生结构性的修改,modCount值会自增。

	public V remove(Object key) {
        Entry<K,V> e = removeEntryForKey(key);
        return (e == null ? null : e.value);
    }
	final Entry<K,V> removeEntryForKey(Object key) {
        if (size == 0) {
            return null;
        }
        int hash = (key == null) ? 0 : hash(key);
        int i = indexFor(hash, table.length);
        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)))) {
                modCount++;
                size--;
                if (prev == e)
                    table[i] = next;
                else
                    prev.next = next;
                e.recordRemoval(this);
                return e;
            }
            prev = e;
            e = next;
        }

        return e;
    }
	final Entry<K,V> removeMapping(Object o) {
        if (size == 0 || !(o instanceof Map.Entry))
            return null;

        Map.Entry<K,V> entry = (Map.Entry<K,V>) o;
        Object key = entry.getKey();
        int hash = (key == null) ? 0 : hash(key);
        int i = indexFor(hash, table.length);
        Entry<K,V> prev = table[i];
        Entry<K,V> e = prev;

        while (e != null) {
            Entry<K,V> next = e.next;
            if (e.hash == hash && e.equals(entry)) {
                modCount++;
                size--;
                if (prev == e)
                    table[i] = next;
                else
                    prev.next = next;
                e.recordRemoval(this);
                return e;
            }
            prev = e;
            e = next;
        }

        return e;
    }
	//在日常编码时快速填充数组可以学习该技巧Arrays.fill
	public void clear() {
        modCount++;
        Arrays.fill(table, null);
        size = 0;
    }

jdk1.7版本的HashMap为什么会产生死锁(链表成环)?

出现问题的场景:

产生死锁或成环的三个条件:

1.产生死锁(链表成环)首先要保证是jdk版本1.7的HashMap

2.多线程环境

3.当addEntry添加元素数量达到一定后,触发扩容机制,扩容时多线程同时操作转移元素会导致死锁成环问题。

扩容时,如果多个线程同时进行转移元素的话,会导致死锁(链表成环)!

演示如下:

然后自己一步步演示作图即可得出成环 死锁的情况! 如下图所示:

ProcessOn Flowchart

解决方法:

Jdk8-扩容:

Java8 HashMap扩容跳过了Jdk7扩容的坑,对源码进行了优化,采用高低位拆分转移方式,避免了链表环的产生。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值