当前位置:   article > 正文

Java多线程 - ReentrantReadWriteLock读写锁详解_threadlocalholdcounter好处

threadlocalholdcounter好处

1. 读写锁简介

现实中有这样一种场景:对共享资源有读和写的操作,且写操作没有读操作那么频繁。在没有写操作的时候,多个线程同时读一个资源没有任何问题,所以应该允许多个线程同时读取共享资源;但是如果一个线程想去写这些共享资源,就不应该允许其他线程对该资源进行读和写的操作了。

针对这种场景,JAVA的并发包提供了读写锁ReentrantReadWriteLock,它表示两个锁,一个是读操作相关的锁,称为共享锁;一个是写相关的锁,称为排他锁,描述如下:

线程进入读锁的前提条件:

没有其他线程的写锁,

没有写请求或者有写请求,但调用线程和持有锁的线程是同一个。

线程进入写锁的前提条件:

没有其他线程的读锁

没有其他线程的写锁

而读写锁有以下三个重要的特性

(1)公平选择性:支持非公平(默认)和公平的锁获取方式,吞吐量还是非公平优于公平。

(2)重进入:读锁和写锁都支持线程重进入。

(3)锁降级:遵循获取写锁、获取读锁再释放写锁的次序,写锁能够降级成为读锁。

2. 源码解读

我们先来看下 ReentrantReadWriteLock 类的整体结构:

  1. public class ReentrantReadWriteLock implements ReadWriteLock, java.io.Serializable {
  2. /** 读锁 */
  3. private final ReentrantReadWriteLock.ReadLock readerLock;
  4. /** 写锁 */
  5. private final ReentrantReadWriteLock.WriteLock writerLock;
  6. final Sync sync;
  7. /** 使用默认(非公平)的排序属性创建一个新的 ReentrantReadWriteLock */
  8. public ReentrantReadWriteLock() {
  9. this(false);
  10. }
  11. /** 使用给定的公平策略创建一个新的 ReentrantReadWriteLock */
  12. public ReentrantReadWriteLock(boolean fair) {
  13. sync = fair ? new FairSync() : new NonfairSync();
  14. readerLock = new ReadLock(this);
  15. writerLock = new WriteLock(this);
  16. }
  17. /** 返回用于写入操作的锁 */
  18. public ReentrantReadWriteLock.WriteLock writeLock() { return writerLock; }
  19. /** 返回用于读取操作的锁 */
  20. public ReentrantReadWriteLock.ReadLock readLock() { return readerLock; }
  21. abstract static class Sync extends AbstractQueuedSynchronizer {}
  22. static final class NonfairSync extends Sync {}
  23. static final class FairSync extends Sync {}
  24. public static class ReadLock implements Lock, java.io.Serializable {}
  25. public static class WriteLock implements Lock, java.io.Serializable {}
  26. }

类的继承关系

  1. public class ReentrantReadWriteLock
  2. implements ReadWriteLock, java.io.Serializable {}

说明:可以看到,ReentrantReadWriteLock实现了ReadWriteLock接口,ReadWriteLock接口定义了获取读锁和写锁的规范,具体需要实现类去实现;同时其还实现了Serializable接口,表示可以进行序列化,在源代码中可以看到ReentrantReadWriteLock实现了自己的序列化逻辑。

类的内部类

ReentrantReadWriteLock有五个内部类,五个内部类之间也是相互关联的。内部类的关系如下图所示。

 

说明:如上图所示,Sync继承自AQS、NonfairSync继承自Sync类、FairSync继承自Sync类(通过构造函数传入的布尔值决定要构造哪一种Sync实例);ReadLock实现了Lock接口、WriteLock也实现了Lock接口。

Sync类:

(1)类的继承关系

abstract static class Sync extends AbstractQueuedSynchronizer {}

说明:Sync抽象类继承自AQS抽象类,Sync类提供了对ReentrantReadWriteLock的支持。

(2)类的内部类

Sync类内部存在两个内部类,分别为HoldCounter和ThreadLocalHoldCounter,其中HoldCounter主要与读锁配套使用,其中,HoldCounter源码如下。

  1. // 计数器
  2. static final class HoldCounter {
  3. // 计数
  4. int count = 0;
  5. // Use id, not reference, to avoid garbage retention
  6. // 获取当前线程的TID属性的值
  7. final long tid = getThreadId(Thread.currentThread());
  8. }

说明:HoldCounter主要有两个属性,count和tid,其中count表示某个读线程重入的次数,tid表示该线程的tid字段的值,该字段可以用来唯一标识一个线程。ThreadLocalHoldCounter的源码如下

  1. // 本地线程计数器
  2. static final class ThreadLocalHoldCounter
  3. extends ThreadLocal<HoldCounter> {
  4. // 重写初始化方法,在没有进行set的情况下,获取的都是该HoldCounter值
  5. public HoldCounter initialValue() {
  6. return new HoldCounter();
  7. }
  8. }

说明:ThreadLocalHoldCounter重写了ThreadLocal的initialValue方法,ThreadLocal类可以将线程与对象相关联。在没有进行set的情况下,get到的均是initialValue方法里面生成的那个HolderCounter对象。

(3)类的属性

  1. abstract static class Sync extends AbstractQueuedSynchronizer {
  2. // 版本序列号
  3. private static final long serialVersionUID = 6317671515068378041L;
  4. // 高16位为读锁,低16位为写锁
  5. static final int SHARED_SHIFT = 16;
  6. // 读锁单位
  7. static final int SHARED_UNIT = (1 << SHARED_SHIFT);
  8. // 读锁最大数量
  9. static final int MAX_COUNT = (1 << SHARED_SHIFT) - 1;
  10. // 写锁最大数量
  11. static final int EXCLUSIVE_MASK = (1 << SHARED_SHIFT) - 1;
  12. // 本地线程计数器
  13. private transient ThreadLocalHoldCounter readHolds;
  14. // 缓存的计数器
  15. private transient HoldCounter cachedHoldCounter;
  16. // 第一个读线程
  17. private transient Thread firstReader = null;
  18. // 第一个读线程的计数
  19. private transient int firstReaderHoldCount;
  20. }

说明:该属性中包括了读锁、写锁线程的最大量。本地线程计数器等。

  1. // 构造函数
  2. Sync() {
  3. // 本地线程计数器
  4. readHolds = new ThreadLocalHoldCounter();
  5. // 设置AQS的状态
  6. setState(getState()); // ensures visibility of readHolds
  7. }

说明:在Sync的构造函数中设置了本地线程计数器和AQS的状态state。

读写状态的设计

同步状态在重入锁的实现中是表示被同一个线程重复获取的次数,即一个整形变量来维护,但是之前的那个表示仅仅表示是否锁定,而不用区分是读锁还是写锁。而读写锁需要在同步状态(一个整形变量)上维护多个读线程和一个写线程的状态。

读写锁对于同步状态的实现是在一个整形变量上通过“按位切割使用”:将变量切割成两部分,高16位表示读,低16位表示写。

http://static.open-open.com/lib/uploadImg/20151031/20151031223319_397.png

假设当前同步状态值为S,get和set的操作如下:

(1)获取写状态:

    S&0x0000FFFF:将高16位全部抹去

(2)获取读状态:

    S>>>16:无符号补0,右移16位

(3)写状态加1:

    S+1

(4)读状态加1:

 S+(1<<16)即S + 0x00010000

在代码层的判断中,如果S不等于0,当写状态(S&0x0000FFFF),而读状态(S>>>16)大于0,则表示该读写锁的读锁已被获取。

写锁的获取与释放

  1. public void lock() {
  2. sync.acquire(1);
  3. }
  4. public void unlock() {
  5. sync.release(1);
  6. }

可以看到就是调用的独占式同步状态的获取与释放,因此真实的实现就是Sync的 tryAcquire和 tryRelease。

写锁的获取,看下tryAcquire:

  1. protected final boolean tryAcquire(int acquires) {
  2. //当前线程
  3. Thread current = Thread.currentThread();
  4. //获取状态
  5. int c = getState();
  6. //写线程数量(即获取独占锁的重入数)
  7. int w = exclusiveCount(c);
  8. //当前同步状态state != 0,说明已经有其他线程获取了读锁或写锁
  9. if (c != 0) {
  10. // 当前state不为0,此时:如果写锁状态为0说明读锁此时被占用返回false;
  11. // 如果写锁状态不为0且写锁没有被当前线程持有返回false
  12. if (w == 0 || current != getExclusiveOwnerThread())
  13. return false;
  14. //判断同一线程获取写锁是否超过最大次数(65535),支持可重入
  15. if (w + exclusiveCount(acquires) > MAX_COUNT)
  16. throw new Error("Maximum lock count exceeded");
  17. //更新状态
  18. //此时当前线程已持有写锁,现在是重入,所以只需要修改锁的数量即可。
  19. setState(c + acquires);
  20. return true;
  21. }
  22. //到这里说明此时c=0,读锁和写锁都没有被获取
  23. //writerShouldBlock表示是否阻塞
  24. if (writerShouldBlock() ||
  25. !compareAndSetState(c, c + acquires))
  26. return false;
  27. //设置锁为当前线程所有
  28. setExclusiveOwnerThread(current);
  29. return true;
  30. }

其中exclusiveCount方法表示占有写锁的线程数量,源码如下:

static int exclusiveCount(int c) { return c & EXCLUSIVE_MASK; }

说明:直接将状态state和(2^16 - 1)做与运算,其等效于将state模上2^16。写锁数量由state的低十六位表示。

从源代码可以看出,获取写锁的步骤如下:

(1)首先获取c、w。c表示当前锁状态;w表示写线程数量。然后判断同步状态state是否为0。如果state!=0,说明已经有其他线程获取了读锁或写锁,执行(2);否则执行(5)。

(2)如果锁状态不为零(c != 0),而写锁的状态为0(w = 0),说明读锁此时被其他线程占用,所以当前线程不能获取写锁,自然返回false。或者锁状态不为零,而写锁的状态也不为0,但是获取写锁的线程不是当前线程,则当前线程也不能获取写锁。

(3)判断当前线程获取写锁是否超过最大次数,若超过,抛异常,反之更新同步状态(此时当前线程已获取写锁,更新是线程安全的),返回true。

(4)如果state为0,此时读锁或写锁都没有被获取,判断是否需要阻塞(公平和非公平方式实现不同),在非公平策略下总是不会被阻塞,在公平策略下会进行判断(判断同步队列中是否有等待时间更长的线程,若存在,则需要被阻塞,否则,无需阻塞),如果不需要阻塞,则CAS更新同步状态,若CAS成功则返回true,失败则说明锁被别的线程抢去了,返回false。如果需要阻塞则也返回false。

(5)成功获取写锁后,将当前线程设置为占有写锁的线程,返回true。

方法流程图如下:

写锁的释放,tryRelease方法:

  1. protected final boolean tryRelease(int releases) {
  2. //若锁的持有者不是当前线程,抛出异常
  3. if (!isHeldExclusively())
  4. throw new IllegalMonitorStateException();
  5. //写锁的新线程数
  6. int nextc = getState() - releases;
  7. //如果独占模式重入数为0了,说明独占模式被释放
  8. boolean free = exclusiveCount(nextc) == 0;
  9. if (free)
  10. //若写锁的新线程数为0,则将锁的持有者设置为null
  11. setExclusiveOwnerThread(null);
  12. //设置写锁的新线程数
  13. //不管独占模式是否被释放,更新独占重入数
  14. setState(nextc);
  15. return free;
  16. }

写锁的释放过程还是相对而言比较简单的:首先查看当前线程是否为写锁的持有者,如果不是抛出异常。然后检查释放后写锁的线程数是否为0,如果为0则表示写锁空闲了,释放锁资源将锁的持有线程设置为null,否则释放仅仅只是一次重入锁而已,并不能将写锁的线程清空。

 说明:此方法用于释放写锁资源,首先会判断该线程是否为独占线程,若不为独占线程,则抛出异常,否则,计算释放资源后的写锁的数量,若为0,表示成功释放,资源不将被占用,否则,表示资源还被占用。其方法流程图如下。

读锁的获取与释放

类似于写锁,读锁的lock和unlock的实际实现对应Sync的 tryAcquireShared 和 tryReleaseShared方法。

读锁的获取,看下tryAcquireShared方法

  1. protected final int tryAcquireShared(int unused) {
  2. // 获取当前线程
  3. Thread current = Thread.currentThread();
  4. // 获取状态
  5. int c = getState();
  6. //如果写锁线程数 != 0 ,且独占锁不是当前线程则返回失败,因为存在锁降级
  7. if (exclusiveCount(c) != 0 &&
  8. getExclusiveOwnerThread() != current)
  9. return -1;
  10. // 读锁数量
  11. int r = sharedCount(c);
  12. /*
  13. * readerShouldBlock():读锁是否需要等待(公平锁原则)
  14. * r < MAX_COUNT:持有线程小于最大数(65535)
  15. * compareAndSetState(c, c + SHARED_UNIT):设置读取锁状态
  16. */
  17. // 读线程是否应该被阻塞、并且小于最大值、并且比较设置成功
  18. if (!readerShouldBlock() &&
  19. r < MAX_COUNT &&
  20. compareAndSetState(c, c + SHARED_UNIT)) {
  21. //r == 0,表示第一个读锁线程,第一个读锁firstRead是不会加入到readHolds中
  22. if (r == 0) { // 读锁数量为0
  23. // 设置第一个读线程
  24. firstReader = current;
  25. // 读线程占用的资源数为1
  26. firstReaderHoldCount = 1;
  27. } else if (firstReader == current) { // 当前线程为第一个读线程,表示第一个读锁线程重入
  28. // 占用资源数加1
  29. firstReaderHoldCount++;
  30. } else { // 读锁数量不为0并且不为当前线程
  31. // 获取计数器
  32. HoldCounter rh = cachedHoldCounter;
  33. // 计数器为空或者计数器的tid不为当前正在运行的线程的tid
  34. if (rh == null || rh.tid != getThreadId(current))
  35. // 获取当前线程对应的计数器
  36. cachedHoldCounter = rh = readHolds.get();
  37. else if (rh.count == 0) // 计数为0
  38. //加入到readHolds中
  39. readHolds.set(rh);
  40. //计数+1
  41. rh.count++;
  42. }
  43. return 1;
  44. }
  45. return fullTryAcquireShared(current);
  46. }

其中sharedCount方法表示占有读锁的线程数量,源码如下:

static int sharedCount(int c)    { return c >>> SHARED_SHIFT; }

说明:直接将state右移16位,就可以得到读锁的线程数量,因为state的高16位表示读锁,对应的第十六位表示写锁数量。

读锁获取锁的过程比写锁稍微复杂些,首先判断写锁是否为0并且当前线程不占有独占锁,直接返回;否则,判断读线程是否需要被阻塞并且读锁数量是否小于最大值并且比较设置状态成功,若当前没有读锁,则设置第一个读线程firstReader和firstReaderHoldCount;若当前线程线程为第一个读线程,则增加firstReaderHoldCount;否则,将设置当前线程对应的HoldCounter对象的值。流程图如下。

注意:更新成功后会在firstReaderHoldCount中或readHolds(ThreadLocal类型的)的本线程副本中记录当前线程重入数(23行至43行代码),这是为了实现jdk1.6中加入的getReadHoldCount()方法的,这个方法能获取当前线程重入共享锁的次数(state中记录的是多个线程的总重入次数),加入了这个方法让代码复杂了不少,但是其原理还是很简单的:如果当前只有一个线程的话,还不需要动用ThreadLocal,直接往firstReaderHoldCount这个成员变量里存重入数,当有第二个线程来的时候,就要动用ThreadLocal变量readHolds了,每个线程拥有自己的副本,用来保存自己的重入数。

fullTryAcquireShared方法:

  1. final int fullTryAcquireShared(Thread current) {
  2. HoldCounter rh = null;
  3. for (;;) { // 无限循环
  4. // 获取状态
  5. int c = getState();
  6. if (exclusiveCount(c) != 0) { // 写线程数量不为0
  7. if (getExclusiveOwnerThread() != current) // 不为当前线程
  8. return -1;
  9. } else if (readerShouldBlock()) { // 写线程数量为0并且读线程被阻塞
  10. // Make sure we're not acquiring read lock reentrantly
  11. if (firstReader == current) { // 当前线程为第一个读线程
  12. // assert firstReaderHoldCount > 0;
  13. } else { // 当前线程不为第一个读线程
  14. if (rh == null) { // 计数器不为空
  15. //
  16. rh = cachedHoldCounter;
  17. if (rh == null || rh.tid != getThreadId(current)) { // 计数器为空或者计数器的tid不为当前正在运行的线程的tid
  18. rh = readHolds.get();
  19. if (rh.count == 0)
  20. readHolds.remove();
  21. }
  22. }
  23. if (rh.count == 0)
  24. return -1;
  25. }
  26. }
  27. if (sharedCount(c) == MAX_COUNT) // 读锁数量为最大值,抛出异常
  28. throw new Error("Maximum lock count exceeded");
  29. if (compareAndSetState(c, c + SHARED_UNIT)) { // 比较并且设置成功
  30. if (sharedCount(c) == 0) { // 读线程数量为0
  31. // 设置第一个读线程
  32. firstReader = current;
  33. //
  34. firstReaderHoldCount = 1;
  35. } else if (firstReader == current) {
  36. firstReaderHoldCount++;
  37. } else {
  38. if (rh == null)
  39. rh = cachedHoldCounter;
  40. if (rh == null || rh.tid != getThreadId(current))
  41. rh = readHolds.get();
  42. else if (rh.count == 0)
  43. readHolds.set(rh);
  44. rh.count++;
  45. cachedHoldCounter = rh; // cache for release
  46. }
  47. return 1;
  48. }
  49. }
  50. }

说明:在tryAcquireShared函数中,如果下列三个条件不满足(读线程是否应该被阻塞、小于最大值、比较设置成功)则会进行fullTryAcquireShared函数中,它用来保证相关操作可以成功。其逻辑与tryAcquireShared逻辑类似,不再累赘。

读锁的释放,tryReleaseShared方法

  1. protected final boolean tryReleaseShared(int unused) {
  2. // 获取当前线程
  3. Thread current = Thread.currentThread();
  4. if (firstReader == current) { // 当前线程为第一个读线程
  5. // assert firstReaderHoldCount > 0;
  6. if (firstReaderHoldCount == 1) // 读线程占用的资源数为1
  7. firstReader = null;
  8. else // 减少占用的资源
  9. firstReaderHoldCount--;
  10. } else { // 当前线程不为第一个读线程
  11. // 获取缓存的计数器
  12. HoldCounter rh = cachedHoldCounter;
  13. if (rh == null || rh.tid != getThreadId(current)) // 计数器为空或者计数器的tid不为当前正在运行的线程的tid
  14. // 获取当前线程对应的计数器
  15. rh = readHolds.get();
  16. // 获取计数
  17. int count = rh.count;
  18. if (count <= 1) { // 计数小于等于1
  19. // 移除
  20. readHolds.remove();
  21. if (count <= 0) // 计数小于等于0,抛出异常
  22. throw unmatchedUnlockException();
  23. }
  24. // 减少计数
  25. --rh.count;
  26. }
  27. for (;;) { // 无限循环
  28. // 获取状态
  29. int c = getState();
  30. // 获取状态
  31. int nextc = c - SHARED_UNIT;
  32. if (compareAndSetState(c, nextc)) // 比较并进行设置
  33. // Releasing the read lock has no effect on readers,
  34. // but it may allow waiting writers to proceed if
  35. // both read and write locks are now free.
  36. return nextc == 0;
  37. }
  38. }

说明:此方法表示读锁线程释放锁。首先判断当前线程是否为第一个读线程firstReader,若是,则判断第一个读线程占有的资源数firstReaderHoldCount是否为1,若是,则设置第一个读线程firstReader为空,否则,将第一个读线程占有的资源数firstReaderHoldCount减1;若当前线程不是第一个读线程,那么首先会获取缓存计数器(上一个读锁线程对应的计数器 ),若计数器为空或者tid不等于当前线程的tid值,则获取当前线程的计数器,如果计数器的计数count小于等于1,则移除当前线程对应的计数器,如果计数器的计数count小于等于0,则抛出异常,之后再减少计数即可。无论何种情况,都会进入无限循环,该循环可以确保成功设置状态state。其流程图如下。

在读锁的获取、释放过程中,总是会有一个对象存在着,同时该对象在获取线程获取读锁是+1,释放读锁时-1,该对象就是HoldCounter。

 要明白HoldCounter就要先明白读锁。前面提过读锁的内在实现机制就是共享锁,对于共享锁其实我们可以稍微的认为它不是一个锁的概念,它更加像一个计数器的概念。一次共享锁操作就相当于一次计数器的操作,获取共享锁计数器+1,释放共享锁计数器-1。只有当线程获取共享锁后才能对共享锁进行释放、重入操作。所以HoldCounter的作用就是当前线程持有共享锁的数量,这个数量必须要与线程绑定在一起,否则操作其他线程锁就会抛出异常。

先看读锁获取锁的部分:

  1. if (r == 0) {//r == 0,表示第一个读锁线程,第一个读锁firstRead是不会加入到readHolds中
  2. firstReader = current;
  3. firstReaderHoldCount = 1;
  4. } else if (firstReader == current) {//第一个读锁线程重入
  5. firstReaderHoldCount++;
  6. } else { //非firstReader计数
  7. HoldCounter rh = cachedHoldCounter;//readHoldCounter缓存
  8. //rh == null 或者 rh.tid != current.getId(),需要获取rh
  9. if (rh == null || rh.tid != current.getId())
  10. cachedHoldCounter = rh = readHolds.get();
  11. else if (rh.count == 0)
  12. readHolds.set(rh); //加入到readHolds中
  13. rh.count++; //计数+1
  14. }

这里为什么要搞一个firstRead、firstReaderHoldCount呢?而不是直接使用else那段代码?这是为了一个效率问题,firstReader是不会放入到readHolds中的,如果读锁仅有一个的情况下就会避免查找readHolds。可能就看这个代码还不是很理解HoldCounter。我们先看firstReader、firstReaderHoldCount的定义:

  1. private transient Thread firstReader = null;
  2. private transient int firstReaderHoldCount;

这两个变量比较简单,一个表示线程,当然该线程是一个特殊的线程,一个是firstReader的重入计数。

HoldCounter的定义:

  1. static final class HoldCounter {
  2. int count = 0;
  3. final long tid = Thread.currentThread().getId();
  4. }

在HoldCounter中仅有count和tid两个变量,其中count代表着计数器,tid是线程的id。但是如果要将一个对象和线程绑定起来仅记录tid肯定不够的,而且HoldCounter根本不能起到绑定对象的作用,只是记录线程tid而已。

诚然,在java中,我们知道如果要将一个线程和对象绑定在一起只有ThreadLocal才能实现。所以如下:

  1. static final class ThreadLocalHoldCounter
  2. extends ThreadLocal<HoldCounter> {
  3. public HoldCounter initialValue() {
  4. return new HoldCounter();
  5. }
  6. }

故而,HoldCounter应该就是绑定线程上的一个计数器,而ThradLocalHoldCounter则是线程绑定的ThreadLocal。从上面我们可以看到ThreadLocal将HoldCounter绑定到当前线程上,同时HoldCounter也持有线程Id,这样在释放锁的时候才能知道ReadWriteLock里面缓存的上一个读取线程(cachedHoldCounter)是否是当前线程。这样做的好处是可以减少ThreadLocal.get()的次数,因为这也是一个耗时操作。需要说明的是这样HoldCounter绑定线程id而不绑定线程对象的原因是避免HoldCounter和ThreadLocal互相绑定而GC难以释放它们(尽管GC能够智能的发现这种引用而回收它们,但是这需要一定的代价),所以其实这样做只是为了帮助GC快速回收对象而已。

3. 总结

通过上面的源码分析,我们可以发现一个现象:

在线程持有读锁的情况下,该线程不能取得写锁(因为获取写锁的时候,如果发现当前的读锁被占用,就马上获取失败,不管读锁是不是被当前线程持有)。

在线程持有写锁的情况下,该线程可以继续获取读锁(获取读锁时如果发现写锁被占用,只有写锁没有被当前线程占用的情况才会获取失败)。

仔细想想,这个设计是合理的:因为当线程获取读锁的时候,可能有其他线程同时也在持有读锁,因此不能把获取读锁的线程“升级”为写锁;而对于获得写锁的线程,它一定独占了读写锁,因此可以继续让它获取读锁,当它同时获取了写锁和读锁后,还可以先释放写锁继续持有读锁,这样一个写锁就“降级”为了读锁。

综上:

一个线程要想同时持有写锁和读锁,必须先获取写锁再获取读锁;写锁可以“降级”为读锁;读锁不能“升级”为写锁。

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

闽ICP备14008679号