JAVA基础知识之多线程——线程通信

传统的线程通信

Object提供了三个方法wait(), notify(), notifyAll()在线程之间进行通信,以此来解决线程间执行顺序等问题。

  • wait():释放当前线程的同步监视控制器,并让当前线程进入阻塞状态,直到别的线程发出notify将该线程唤醒。
  • notify():唤醒在等待控制监视器的其中一个线程(随机)。只有当前线程释放了同步监视器锁(调用wait)之后,被唤醒的线程才有机会执行。
  • notifyAll():与上面notify的区别是同时唤醒多个等待线程。

值得注意的是这三个方法是属于Object而不是属于Thread的,但是调用的时候必须用同步监视器来调用,

  • 对于synchronized修饰的同步方法,由于方法所在类对象(this)就是同步监视器,因此可以直接在同步方法中调用这三个方法;
  • 对于同步代码块,synchronized(obj) { ... },则需要用空号钟的obj来调用。

生产者-消费者问题模型

在经典的生产者-消费者问题中,需要使用线程通信来解决。

假设有这么一个场景,有一个线程需要存钱进一个账户,有多个线程需要从这个账户取钱,要求是每次必须先存钱之后才能取钱,而且取钱之后必须存钱,

存钱和取钱不能同时发生两次,而是要保持顺序不变,如何实现这个需求呢。

下面是用同步方法结合线程通信的方式来实现的思路,

  • 首先在Account类中定义两个同步方法,deposit和draw用来确保存款和取款操作的原子性。
  • 在Account类中定义用标识符flag, 由deposit和draw共用。初始值为false,表示只能存款。 如果为false,表示只能取款。
  • 定义一个存款线程类,去调用Account类的同步方法deposit,在deposit中先对flag进行判断,如果不为false,则调用wait阻塞存款线程,等待取款线程发出notice。存款完成之后,将flag改为true.
  • 定义一个取款线程类,去调用Account类的同步方法draw,在draw中先对flag进行判断,如果不为true,则调用wait阻塞取款线程,等待存线程发出notice。取款完成之后,将flag改为false.
  • 定义测试类,同时启动一个(或多个)存款线程进行存款,同时启动多个取款线程去取款,存款(取款)线程之间不会有先后顺序,但是存款和取款直接会有严格的先后顺序,这就解决了生产者消费者问题

下面给出实现代码

在Account类中定义两个同步方法,draw和deposit

 1 package threads.sync;
 2 
 3 public class Account {
 4     private String accountNo;
 5     private double balance;
 6     //if flag = false, means only deposit can be done
 7     private boolean flag = false;
 8     public Account() {}
 9     
10     
11     public Account(String accountNo, double balance) {
12         this.accountNo = accountNo;
13         this.balance = balance;
14     }
15 
16     public String getAccountNo() {
17         return accountNo;
18     }
19 
20     public void setAccountNo(String accountNo) {
21         this.accountNo = accountNo;
22     }
23 
24     public double getBalance() {
25         return balance;
26     }
27 
28     public void setBalance(double balance) {
29         this.balance = balance;
30     }
31     
32     
33     public int hashCode() {
34         return accountNo.hashCode();
35     }
36     
37     public boolean equals(Object obj) {
38         if (this == obj) return true;
39         if (obj != null && obj.getClass() == Account.class) {
40             Account target = (Account)obj;
41             return target.getAccountNo().equals(accountNo);
42         }
43         return false;
44     }
45     
46     public synchronized void draw(double drawAmount) {
47         try {
48             //if flag = false, means only deposit can be done, draw method will be blocked
49             if (!flag) {
50                 wait();
51             } else {
52                 System.out.println(Thread.currentThread().getName()
53                         + "        draw money: " + drawAmount);
54                 balance -= drawAmount;
55                 System.out.println("            "
56                         + " balance : " + balance);
57                 flag = false;
58                 notifyAll();
59             } 
60         } catch (InterruptedException ex) {
61             ex.printStackTrace();
62         }
63     }
64     
65     public synchronized void deposit(double depositAmount) {
66         try {
67             //if flag = false, means only draw can be done, deposit method will be blocked
68             if (flag) {
69                 wait();
70             } else {
71                 System.out.println(Thread.currentThread().getName()
72                         + "        deposit money: " + depositAmount);
73                 balance += depositAmount;
74                 System.out.println("            "
75                         + " balance : " + balance);
76                 flag = true;
77                 notifyAll();
78             } 
79         } catch (InterruptedException ex) {
80             ex.printStackTrace();
81         }    
82     }
83     
84 }

定义一个存款线程类depositThread

 1 package threads.sync;
 2 
 3 public class DepositThread extends Thread {
 4     private Account account;
 5     private double depositAmount;
 6     public DepositThread(String name, Account account, double depositAmount) {
 7         super(name);
 8         this.setAccount(account);
 9         this.setDepositAmount(depositAmount);
10     }
11     public Account getAccount() {
12         return account;
13     }
14     public void setAccount(Account account) {
15         this.account = account;
16     }
17     
18     public double getDepositAmount() {
19         return depositAmount;
20     }
21     public void setDepositAmount(double depositAmount) {
22         this.depositAmount = depositAmount;
23     }
24     
25     public void run() {
26         for(int i=0 ; i<10; i++) {
27             account.deposit(depositAmount);
28         }
29 
30     }
31 
32 }

定义一个取款线程类depositThread

 1 package threads.sync;
 2 
 3 public class DrawThread extends Thread {
 4     private Account account;
 5     private double drawAmount;
 6     public DrawThread(String name, Account account, double drawAmount) {
 7         super(name);
 8         this.setAccount(account);
 9         this.setDrawAmount(drawAmount);
10     }
11     public Account getAccount() {
12         return account;
13     }
14     public void setAccount(Account account) {
15         this.account = account;
16     }
17     public double getDrawAmount() {
18         return drawAmount;
19     }
20     public void setDrawAmount(double drawAmount) {
21         this.drawAmount = drawAmount;
22     }
23     
24     public void run() {
25         for(int i=0 ; i<10; i++) {
26             account.draw(drawAmount);
27         }
28     }
29 }

下面是测试类,存款线程中会有10次存款,三个取款线程中总共会有30次取款,

 1 package threads.sync;
 2 
 3 public class DrawTest {
 4     public static void main(String[] args) {
 5         Account acc = new Account("123456",1000);
 6         new DrawThread("DrawThread", acc, 800).start();
 7         new DepositThread("DepositThread-A",acc,800).start();
 8         new DepositThread("DepositThread-B",acc,800).start();
 9         new DepositThread("DepositThread-C",acc,800).start();
10     }
11 }

执行结果,

 1 DepositThread-A        deposit money: 800.0
 2                 balance : 1800.0
 3 DrawThread        draw money: 800.0
 4                 balance : 1000.0
 5 DepositThread-B        deposit money: 800.0
 6                 balance : 1800.0
 7 DrawThread        draw money: 800.0
 8                 balance : 1000.0
 9 DepositThread-C        deposit money: 800.0
10                 balance : 1800.0
11 DrawThread        draw money: 800.0
12                 balance : 1000.0
13 DepositThread-C        deposit money: 800.0
14                 balance : 1800.0
15 DrawThread        draw money: 800.0
16                 balance : 1000.0
17 DepositThread-C        deposit money: 800.0
18                 balance : 1800.0
19 DrawThread        draw money: 800.0
20                 balance : 1000.0
21 DepositThread-C        deposit money: 800.0
22                 balance : 1800.0
23 DrawThread        draw money: 800.0
24                 balance : 1000.0
25 DepositThread-C        deposit money: 800.0
26                 balance : 1800.0
27 DrawThread        draw money: 800.0
28                 balance : 1000.0
29 DepositThread-A        deposit money: 800.0
30                 balance : 1800.0

从执行结果中可以看到,三个取款线程ABC执行顺序随机,但是总是在存款完成后,才会进行取款操作,而且无论存款还是取款,都不会同时进行两次。

使用condition控制线程通信

如果程序使用lock来同步线程的话,就要使用condition来进行线程通信。

在lock同步线程中,lock 对象就是一个显示的同步监视器,但是这个显示的同步监视器不直接阻塞或者通知线程,而是通过condition——lock对象通过调用newCondition方法返回一个与lock关联的condition对象,由condition对象来控制线程阻塞(await)和发出信号(single)唤醒其他线程。

与synchronized同步线程方式对应的是,conditions方式也提供了三个方法,

await:类似于synchronized隐式同步控制器对象调用的wait方法,可以阻塞当前线程,直到在别的线程中调用了condition的singal方法唤醒该线程。

signal:随机唤醒一个被await阻塞的线程。注意只有在当前线程已经释放lock同步监视器之后,被唤醒的其他线程才有机会执行。

signalAll:与上面类似,但是是唤醒所有线程。

下面用condition的方式来实现前面的银行取钱的例子,只需要修改Account类,改用lock同步线程,condition线程通信,

  1 package threads.sync;
  2 
  3 import java.util.concurrent.locks.Condition;
  4 import java.util.concurrent.locks.Lock;
  5 import java.util.concurrent.locks.ReentrantLock;
  6 
  7 public class Account {
  8     private String accountNo;
  9     private double balance;
 10     private boolean flag = false;
 11     //显示定义lock对象
 12     private final Lock lock = new ReentrantLock();
 13     //获取lock对象对应的condition
 14     private final Condition cond = lock.newCondition();
 15     public Account() {}
 16     
 17     
 18     public Account(String accountNo, double balance) {
 19         this.accountNo = accountNo;
 20         this.balance = balance;
 21     }
 22 
 23     public String getAccountNo() {
 24         return accountNo;
 25     }
 26 
 27     public void setAccountNo(String accountNo) {
 28         this.accountNo = accountNo;
 29     }
 30 
 31     public double getBalance() {
 32         return balance;
 33     }
 34 
 35     public void setBalance(double balance) {
 36         this.balance = balance;
 37     }
 38     
 39     
 40     public int hashCode() {
 41         return accountNo.hashCode();
 42     }
 43     
 44     public boolean equals(Object obj) {
 45         if (this == obj) return true;
 46         if (obj != null && obj.getClass() == Account.class) {
 47             Account target = (Account)obj;
 48             return target.getAccountNo().equals(accountNo);
 49         }
 50         return false;
 51     }
 52     
 53     public void draw(double drawAmount) {
 54         lock.lock();
 55         try {
 56             //if flag = false, means only deposit can be done, draw method will be blocked
 57             if (!flag) {
 58                 //this.wait();
 59                 cond.await();
 60             } else {
 61                 System.out.println(Thread.currentThread().getName()
 62                         + "        draw money: " + drawAmount);
 63                 balance -= drawAmount;
 64                 System.out.println("            "
 65                         + "    balance : " + balance);
 66                 flag = false;
 67                 //this.notifyAll();
 68                 cond.signalAll();
 69             } 
 70         } catch (InterruptedException ex) {
 71             ex.printStackTrace();
 72         } finally {
 73             lock.unlock();
 74         }
 75     }
 76     
 77     public void deposit(double depositAmount) {
 78         lock.lock();
 79         try {
 80             //if flag = false, means only draw can be done, deposit method will be blocked
 81             if (flag) {
 82                 //this.wait();
 83                 cond.await();
 84             } else {
 85                 System.out.println(Thread.currentThread().getName()
 86                         + "        deposit money: " + depositAmount);
 87                 balance += depositAmount;
 88                 System.out.println("            "
 89                         + "    balance : " + balance);
 90                 flag = true;
 91                 //this.notifyAll();
 92                 cond.signalAll();
 93             } 
 94         } catch (InterruptedException ex) {
 95             ex.printStackTrace();
 96         } finally {
 97             lock.unlock();
 98         }
 99     }
100     
101 }

对比用synchronized方式同步线程的例子,前面例子中是隐式的同步监视器(this)调用wait和notify来通信,

而本例是显示同步监视器(lock)的关联对象(condition)调用await和signal来通信,执行结果与前面一样不再给出。

使用阻塞队列(BlockingQueue)控制线程通信

BlockingQueue是JAVA5提供的一个队列接口,但这个队列并不是用作一个容器,而是作为线程的同步工具。

它可以很好地解决生产者消费者问题,而且比前面提到的两种方式更为灵活,

BlockingQueue的特征是,

当生产者线程试图向BlockingQueue存入元素时,如果队列已满,生产者线程将会阻塞,

当消费者线程试图从BlockingQueue取出元素时,如果队列为空,消费者线程将会阻塞

对比前面线程通信的例子,synchronized同步方法/代码块和lock+condition方式中,都只能控制生产者和消费者按固定顺序执行,

但BlockingQueue则是可以通过集合中的元素个数(商品数量)来控制线程执行顺序,通过调整集合容量可以控制线程切换的条件。

集合(商品)为空时,消费者阻塞,只能执行生产者线程;集合(商品)已满时,生产者阻塞,只能执行消费者线程。

BlockingQueue接口有很多实现类,下面演示最常用的实现类ArrayBlockQueue控制线程通信,

定义一个生产者线程类Producter

 1 package threads.sync;
 2 
 3 import java.util.concurrent.BlockingQueue;
 4 
 5 public class Producter extends Thread {
 6     private BlockingQueue<String> bq;
 7     
 8     public Producter(BlockingQueue<String> bq) {
 9         this.bq = bq;
10     }
11     
12     public void run() {
13         String[] strArr = new String[] {
14                 "Java",
15                 "Struts",
16                 "Spring"
17         };
18         
19         for(int i = 0; i<999999; i++) {
20             System.out.println(getName()+" 生产者准备生产集合元素");
21             try {
22                 Thread.sleep(200);
23                 //如果队列已满,线程将阻塞
24                 bq.put(strArr[i % 3]);
25             } catch (InterruptedException e) {
26                 e.printStackTrace();
27             }
28             System.out.println(getName()+" 生产完成: " + bq);
29         }
30     }
31 }

 

定义一个消费者类Consumer

 1 package threads.sync;
 2 
 3 import java.util.concurrent.BlockingQueue;
 4 
 5 public class Consumer extends Thread {
 6     private BlockingQueue<String> bq;
 7     
 8     public Consumer(BlockingQueue<String> bq) {
 9         this.bq = bq;
10     }
11     
12     public void run() {
13         
14         while (true) {
15             System.out.println(getName()+" 消费者准备消费集合元素");
16             try {
17                 Thread.sleep(200);
18                 //如果队列已空,线程将阻塞
19                 bq.take();
20             } catch (InterruptedException e) {
21                 e.printStackTrace();
22             }
23             System.out.println(getName()+" 消费完成: " + bq);
24         }
25     }
26 }

 

在测试类中,定义一个容量为2的阻塞集合,

启动三个生产者线程, 每个线程都在不停生产商品,存入阻塞队列中,

启动一个消费者线程,每个线程也在不停从阻塞队列中取出商品,

 1 package threads.sync;
 2 
 3 import java.util.concurrent.ArrayBlockingQueue;
 4 import java.util.concurrent.BlockingQueue;
 5 
 6 public class BlockingQueueTest {
 7     public static void main(String[] args) {
 8         BlockingQueue<String> bq = new ArrayBlockingQueue<String>(2);
 9         new Producter(bq).start();
10         new Producter(bq).start();
11         new Producter(bq).start();
12         new Consumer(bq).start();
13     }
14 }

 

执行结果,从执行结果中可以看到,只要集合中有元素且集合没有满,那么生产者和消费者线程都有机会得到执行,具体谁有机会要看谁抢到CPU执行片,

但是当集合空了的时候,例如第7行(Thread-8 消费完成: []),接着又有一个消费者线程执行,但是因此集合为空而阻塞了,此时只有生产者线程能执行,

当集合满了的时候,例如第11行(Thread-7 生产完成: [Java, Java]),接着又有一个生产者线程执行,但是因为集合已满而阻塞了,此时只有消费者线程能执行。

 1 Thread-5 生产者准备生产集合元素
 2 Thread-6 生产者准备生产集合元素
 3 Thread-7 生产者准备生产集合元素
 4 Thread-8 消费者准备消费集合元素
 5 Thread-5 生产完成: [Java]
 6 Thread-5 生产者准备生产集合元素
 7 Thread-8 消费完成: []
 8 Thread-8 消费者准备消费集合元素
 9 Thread-6 生产完成: [Java]
10 Thread-6 生产者准备生产集合元素
11 Thread-7 生产完成: [Java, Java]
12 Thread-7 生产者准备生产集合元素
13 Thread-8 消费完成: [Java]
14 Thread-8 消费者准备消费集合元素
15 Thread-5 生产完成: [Java, Struts]
16 Thread-5 生产者准备生产集合元素
17 Thread-8 消费完成: [Struts]
18 Thread-8 消费者准备消费集合元素
19 Thread-6 生产完成: [Struts, Struts]
20 Thread-6 生产者准备生产集合元素
21 Thread-8 消费完成: [Struts]
22 Thread-8 消费者准备消费集合元素
23 Thread-7 生产完成: [Struts, Struts]
24 Thread-7 生产者准备生产集合元素
25 Thread-8 消费完成: [Struts]
26 Thread-8 消费者准备消费集合元素
27 Thread-5 生产完成: [Struts, Spring]
28 Thread-5 生产者准备生产集合元素
29 Thread-8 消费完成: [Spring]
30 Thread-8 消费者准备消费集合元素
31 Thread-6 生产完成: [Spring, Spring]
32 Thread-6 生产者准备生产集合元素
33 Thread-8 消费完成: [Spring]
34 Thread-8 消费者准备消费集合元素

 

上面的例子来自李刚的疯狂JAVA, 但个人认为并不是太好,因为无论在生产者还是消费者线程中,打印bq操作前后的两段日志并不是原子操作,这会导致打印的日志不准确,

例如下面的运行结果,从第5行看到(Thread-6 生产完成: []),刚刚执行完一个生产者线程中的入队操作,但是打印队列却是空的,原因就在于在打印两行日志期间,消费者线做了取元素的操作。

 1 Thread-6 生产者准备生产集合元素
 2 Thread-5 生产者准备生产集合元素
 3 Thread-7 生产者准备生产集合元素
 4 Thread-8 消费者准备消费集合元素
 5 Thread-6 生产完成: []
 6 Thread-6 生产者准备生产集合元素
 7 Thread-5 生产完成: [Java]
 8 Thread-5 生产者准备生产集合元素
 9 Thread-7 生产完成: [Java, Java]
10 Thread-7 生产者准备生产集合元素
11 Thread-8 消费完成: []
12 Thread-8 消费者准备消费集合元素
13 Thread-7 生产完成: [Java, Struts]
14 Thread-7 生产者准备生产集合元素
15 Thread-8 消费完成: [Java, Struts]
16 Thread-8 消费者准备消费集合元素
17 Thread-8 消费完成: [Struts]
18 Thread-5 生产完成: [Struts, Struts]
19 Thread-5 生产者准备生产集合元素
20 Thread-8 消费者准备消费集合元素
21 Thread-8 消费完成: [Struts]
22 Thread-8 消费者准备消费集合元素
23 Thread-6 生产完成: [Struts, Struts]
24 Thread-6 生产者准备生产集合元素
25 Thread-8 消费完成: [Struts]
26 Thread-8 消费者准备消费集合元素
27 Thread-6 生产完成: [Struts, Spring]
28 Thread-6 生产者准备生产集合元素
29 Thread-8 消费完成: [Spring]
30 Thread-8 消费者准备消费集合元素
31 Thread-5 生产完成: [Spring, Spring]
32 Thread-5 生产者准备生产集合元素
33 Thread-8 消费完成: [Spring]
34 Thread-8 消费者准备消费集合元素
35 Thread-5 生产完成: [Spring, Java]
36 Thread-5 生产者准备生产集合元素
37 Thread-8 消费完成: [Java]
38 Thread-8 消费者准备消费集合元素
39 Thread-6 生产完成: [Java, Java]
40 Thread-6 生产者准备生产集合元素
41 Thread-8 消费完成: [Java]
42 Thread-8 消费者准备消费集合元素
43 Thread-7 生产完成: [Java, Spring]
44 Thread-7 生产者准备生产集合元素
45 Thread-8 消费完成: [Spring]
46 Thread-8 消费者准备消费集合元素
47 Thread-7 生产完成: [Spring, Java]
48 Thread-7 生产者准备生产集合元素
49 Thread-8 消费完成: [Java]
50 Thread-8 消费者准备消费集合元素
51 Thread-7 生产完成: [Java, Struts]
52 Thread-7 生产者准备生产集合元素
53 Thread-8 消费完成: [Struts]
54 Thread-8 消费者准备消费集合元素
55 Thread-7 生产完成: [Struts, Spring]
56 Thread-7 生产者准备生产集合元素
57 Thread-8 消费完成: [Spring]
58 Thread-8 消费者准备消费集合元素
59 Thread-7 生产完成: [Spring, Java]
60 Thread-7 生产者准备生产集合元素
61 Thread-8 消费完成: [Java]
62 Thread-8 消费者准备消费集合元素
63 Thread-7 生产完成: [Java, Struts]
64 Thread-7 生产者准备生产集合元素
65 Thread-7 生产完成: [Struts, Spring]
66 Thread-7 生产者准备生产集合元素
67 Thread-8 消费完成: [Struts, Spring]
68 Thread-8 消费者准备消费集合元素
69 Thread-8 消费完成: [Spring]
70 Thread-8 消费者准备消费集合元素
71 Thread-5 生产完成: [Spring, Struts]
72 Thread-5 生产者准备生产集合元素
73 Thread-8 消费完成: [Struts]
74 Thread-8 消费者准备消费集合元素
75 Thread-6 生产完成: [Struts, Struts]
76 Thread-6 生产者准备生产集合元素
77 Thread-8 消费完成: [Struts, Spring]
78 Thread-8 消费者准备消费集合元素
79 Thread-6 生产完成: [Struts, Spring]
80 Thread-6 生产者准备生产集合元素

 

转载于:https://www.cnblogs.com/fysola/p/6072883.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
使用Python绘制混淆矩阵,可以使用sklearn.metrics包中的confusion_matrix函数。首先,需要将预测结果和真实标签以类似的格式赋值给y_pred和y_true变量。然后,可以使用confusion_matrix函数生成混淆矩阵C,可以通过labels参数指定类别的标签。接下来,可以使用matplotlib.pyplot中的函数绘制矩阵图,使用plt.matshow(C, cmap=plt.cm.Reds)来展示混淆矩阵的颜色。可以使用plt.annotate函数在矩阵图中显示每个元素的值。最后,可以使用plt.xlabel和plt.ylabel函数设置x轴和y轴的标签。最后,使用plt.show函数显示绘制好的混淆矩阵图。 [1 [2] 示例代码如下: ```python from sklearn.metrics import confusion_matrix import matplotlib.pyplot as plt y_pred = [] # 预测结果 y_true = [] # 真实标签 C = confusion_matrix(y_true, y_pred, labels=['0','1','2','3','4']) plt.matshow(C, cmap=plt.cm.Reds) for i in range(len(C)): for j in range(len(C)): plt.annotate(C[j, i], xy=(i, j), horizontalalignment='center', verticalalignment='center') plt.ylabel('True label') plt.xlabel('Predicted label') plt.show() ``` 这段代码会根据给定的预测结果和真实标签生成混淆矩阵,并使用矩阵图展示混淆矩阵的颜色。每个元素表示预测为某个类别的样本数量。通过调整代码中的参数和标签,可以根据不同的需求进行自定义。 [1 [2 [3<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [详解使用python绘制混淆矩阵confusion_matrix)](https://download.csdn.net/download/weixin_38580959/12861679)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT0_1"}}] [.reference_item style="max-width: 50%"] - *2* *3* [利用python绘制混淆矩阵](https://blog.csdn.net/weixin_43818631/article/details/121309660)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT0_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值