Java锁有关的面试题和Java锁的例子详解 MySQL锁机制的基本概念、实现方式、使用场景以及性能影响。 MySQL面试题和例子详解 python锁机制有关的面试题和python锁的例子详解

2 篇文章 0 订阅
2 篇文章 0 订阅

目录

Java 锁机制 

1. 乐观锁(Optimistic Locking)

2. 悲观锁(Pessimistic Locking)

3. 自旋锁(Spin Lock)

4. Synchronized 同步锁

java 锁机制   示例代码  详解

乐观锁示例:

悲观锁示例:

自旋锁示例:

Synchronized 同步锁示例:

MySQL的锁机制面试题

MySQL锁机制的基本概念、实现方式、使用场景以及性能影响等方面

1. 什么是MySQL锁机制?

2. MySQL有哪些类型的锁?

3. 什么是共享锁和排他锁?

4. InnoDB存储引擎支持哪些锁?

5. 什么是意向锁?

6. 什么是死锁?如何解决死锁?

7. 如何查看当前MySQL的锁信息?

8. 什么是MVCC(多版本并发控制)?

9. 什么是乐观锁和悲观锁?MySQL中如何实现?

10. 如何使用SELECT ... FOR UPDATE和LOCK IN SHARE MODE?

11. 什么是Gap锁?

12. 什么是Next-Key锁?

13. 如何避免锁升级?

14. 如何使用MySQL的锁来实现并发控制?

15. MySQL的锁机制对性能有什么影响?

MySQL锁机制的例子详解

1. 锁的类型

1.1 表锁(Table Lock)

1.2 行锁(Row Lock)

1.3 页锁(Page Lock)

2. 锁的模式

2.1 共享锁(S Lock)

2.2 排他锁(X Lock)

3. 意向锁(Intention Lock)

4. 死锁与解决

4.1 死锁

4.2 解决方法

5. MVCC(多版本并发控制)

6. Gap锁和Next-Key锁

6.1 Gap锁

6.2 Next-Key锁

示例代码

7. 查看锁信息

8. 性能影响与优化

Python锁机制面试题

1. 什么是Python中的GIL(全局解释器锁)?

2. Python中的锁类型有哪些?

3. 什么是threading.Lock?

4. 什么是threading.RLock?与threading.Lock有何不同?

5. 如何使用threading.Condition实现线程间的通信?

6. 如何使用threading.Semaphore控制对共享资源的访问?

7. 什么是死锁?如何避免死锁?

8. 如何使用threading.Event实现线程间的信号通信?

9. 什么是threading.Barrier?如何使用?

10. 如何使用threading.Lock实现线程安全的计数器?

python的锁机制 详解

1. GIL(全局解释器锁)

2. 锁的类型

2.1 threading.Lock

2.2 threading.RLock

2.3 threading.Condition

2.4 threading.Semaphore

2.5 threading.Event

2.6 threading.Barrier

3. 死锁与避免

4. 线程安全的计数器

5. 性能影响与优化


写文不易,给我点点赞和收藏吧

Java 锁机制 

Java 提供了多种锁机制来处理并发编程中的同步问题。以下是几种常见的锁机制及其原理和使用场景:

1. 乐观锁(Optimistic Locking)

原理:

  • 乐观锁假设并发冲突的可能性较低,因此在读取数据时不加锁。
  • 在更新数据时,乐观锁会检查数据是否被其他线程修改过。通常通过版本号(version)或时间戳(timestamp)来实现。
  • 如果版本号一致,则进行更新;否则,重试读取-比较-写入的过程。

实现:

  • Java 中的乐观锁通常通过 CAS(Compare-And-Swap)操作实现。CAS 是一种原子操作,比较当前值与预期值,如果相同则更新,否则失败。

适用场景:

  • 适用于读多写少的场景,减少锁的开销。
2. 悲观锁(Pessimistic Locking)

原理:

  • 悲观锁假设并发冲突的可能性较高,因此在读取和写入数据时都加锁。
  • 其他线程在尝试访问被锁定的数据时会被阻塞,直到锁被释放。

实现:

  • Java 中的悲观锁可以通过 synchronized 关键字或 ReentrantLock 实现。
  • ReentrantLock 是基于 AQS(AbstractQueuedSynchronizer)框架的锁,先尝试使用 CAS 乐观锁获取锁,获取不到则转换为悲观锁。

适用场景:

  • 适用于写操作频繁的场景,确保数据的一致性。
3. 自旋锁(Spin Lock)

原理:

  • 自旋锁在等待锁释放时不会阻塞线程,而是让线程在循环中不断尝试获取锁。
  • 如果持有锁的线程能在短时间内释放锁,自旋锁可以避免线程阻塞和上下文切换的开销。

优缺点:

  • 优点:减少线程阻塞和上下文切换的开销,适用于锁竞争不激烈且锁持有时间短的场景。
  • 缺点:自旋会消耗 CPU 资源,如果锁竞争激烈或持有时间长,自旋锁的消耗可能大于线程阻塞的开销。

实现:

  • JDK 1.6 引入了适应性自旋锁,自动调整自旋时间。
  • 可以通过 JVM 参数 -XX:+UseSpinning 和 -XX:PreBlockSpin 控制自旋锁的行为(JDK 1.7 后由 JVM 自动控制)。
4. Synchronized 同步锁

原理:

  • synchronized 关键字可以锁住任意非 NULL 对象,属于独占式悲观锁和可重入锁。
  • 作用于方法时,锁住的是对象实例(this)。
  • 作用于静态方法时,锁住的是类的 Class 对象,相当于全局锁。
  • 作用于代码块时,锁住的是指定的对象。

核心组件:

  • Wait Set:存放调用 wait 方法被阻塞的线程。
  • Contention List:竞争队列,所有请求锁的线程首先被放在这里。
  • Entry List:有资格成为候选资源的线程从竞争队列移动到这里。
  • OnDeck:当前正在竞争锁资源的线程。
  • Owner:当前持有锁的线程。
  • !Owner:当前释放锁的线程。

适用场景:

  • 适用于需要确保线程安全的场景,synchronized 简单易用,但可能会导致性能瓶颈。

java 锁机制   示例代码  详解

以下是一些示例代码,展示如何使用这些锁机制:

乐观锁示例:

 
import java.util.concurrent.atomic.AtomicInteger;

public class OptimisticLockExample {
    private AtomicInteger version = new AtomicInteger(0);

    public void updateData() {
        int currentVersion;
        do {
            currentVersion = version.get();
            // 读取数据并进行操作
        } while (!version.compareAndSet(currentVersion, currentVersion + 1));
        // 更新数据
    }
}

悲观锁示例:

public class PessimisticLockExample {
    private final Object lock = new Object();

    public void updateData() {
        synchronized (lock) {
            // 读取和更新数据
        }
    }
}

自旋锁示例:

public class SpinLockExample {
    private volatile boolean isLocked = false;

    public void lock() {
        while (!compareAndSwap(false, true)) {
            // 自旋等待
        }
    }

    public void unlock() {
        isLocked = false;
    }

    private boolean compareAndSwap(boolean expected, boolean newValue) {
        if (isLocked == expected) {
            isLocked = newValue;
            return true;
        }
        return false;
    }
}

Synchronized 同步锁示例:

public class SynchronizedExample {
    public synchronized void synchronizedMethod() {
        // 线程安全的方法
    }

    public void synchronizedBlock() {
        synchronized (this) {
            // 线程安全的代码块
        }
    }

    public static synchronized void staticSynchronizedMethod() {
        // 静态方法的线程安全
    }
}

这些示例展示了不同锁机制的基本用法,根据具体的应用场景选择合适的锁机制可以有效提高并发程序的性能和可靠性。

MySQL的锁机制面试题

MySQL锁机制的基本概念、实现方式、使用场景以及性能影响等方面

1. 什么是MySQL锁机制?

回答
MySQL锁机制是为了保证数据的一致性和完整性而设计的。它通过控制并发访问来防止多个事务同时修改同一数据,从而避免数据冲突和不一致。

2. MySQL有哪些类型的锁?

回答
MySQL主要有以下几种类型的锁:

  • 表锁(Table Lock):锁住整个表,适用于全表扫描或大批量数据更新操作。
  • 行锁(Row Lock):锁住特定的行,适用于高并发环境下的细粒度控制。
  • 页锁(Page Lock):锁住特定的页,介于表锁和行锁之间。

3. 什么是共享锁和排他锁?

回答

  • 共享锁(S Lock):允许多个事务同时读取数据,但不能修改数据。
  • 排他锁(X Lock):只允许一个事务读取和修改数据,其他事务不能访问。

4. InnoDB存储引擎支持哪些锁?

回答
InnoDB存储引擎支持行级锁和表级锁。行级锁包括共享锁和排他锁,表级锁包括意向共享锁(IS)和意向排他锁(IX)。

5. 什么是意向锁?

回答
意向锁是一种表级锁,用于表示某个事务即将在表中的某些行上加行级锁。意向锁分为意向共享锁(IS)和意向排他锁(IX),它们用于提高锁的效率,避免全表扫描。

6. 什么是死锁?如何解决死锁?

回答
死锁是指两个或多个事务相互等待对方持有的锁,从而导致事务无法继续执行。
解决方法

  • 死锁检测:InnoDB存储引擎会自动检测死锁,并回滚其中一个事务。
  • 超时机制:设置事务等待锁的超时时间,超过时间自动回滚。
  • 合理设计事务:尽量减少锁的持有时间,避免长时间持有锁。

7. 如何查看当前MySQL的锁信息?

回答
可以通过以下SQL语句查看当前的锁信息:

CopySHOW ENGINE INNODB STATUS;

或者查询information_schema中的锁相关表:

CopySELECT * FROM information_schema.INNODB_LOCKS;
SELECT * FROM information_schema.INNODB_LOCK_WAITS;

8. 什么是MVCC(多版本并发控制)?

回答
MVCC(Multi-Version Concurrency Control)是一种并发控制机制,通过保存数据的多个版本来实现读写并发。InnoDB存储引擎使用MVCC来实现快照读,从而避免读写冲突,提高并发性能。

9. 什么是乐观锁和悲观锁?MySQL中如何实现?

回答

  • 乐观锁:假设并发冲突的可能性较低,在提交数据时检查是否有冲突。可以通过版本号或时间戳实现。
  • 悲观锁:假设并发冲突的可能性较高,在读取数据时加锁,防止其他事务修改。可以通过SELECT ... FOR UPDATE实现。

10. 如何使用SELECT ... FOR UPDATELOCK IN SHARE MODE

回答

  • SELECT ... FOR UPDATE:加排他锁,防止其他事务修改数据。
CopySELECT * FROM table_name WHERE condition FOR UPDATE;
  • LOCK IN SHARE MODE:加共享锁,允许其他事务读取但不允许修改数据。
CopySELECT * FROM table_name WHERE condition LOCK IN SHARE MODE;

11. 什么是Gap锁?

回答
Gap锁(间隙锁)是InnoDB存储引擎的一种行级锁,用于锁定一个范围内的所有记录,但不包括边界值。Gap锁用于防止幻读现象,确保事务的一致性。

12. 什么是Next-Key锁?

回答
Next-Key锁是InnoDB存储引擎的一种行级锁,它是行锁和Gap锁的结合体,锁定一个范围内的所有记录,包括边界值。Next-Key锁用于防止幻读现象,确保事务的一致性。

13. 如何避免锁升级?

回答
锁升级是指从行级锁升级为表级锁,通常是由于锁的数量过多导致的。避免锁升级的方法包括:

  • 优化SQL查询,减少锁的数量。
  • 合理设计索引,避免全表扫描。
  • 控制事务的粒度,尽量减少锁的持有时间。

14. 如何使用MySQL的锁来实现并发控制?

回答
可以通过以下几种方式使用MySQL的锁来实现并发控制:

  • 使用SELECT ... FOR UPDATELOCK IN SHARE MODE来控制读写并发。
  • 使用事务和锁机制来确保数据的一致性和完整性。
  • 使用乐观锁和悲观锁来处理并发冲突。

15. MySQL的锁机制对性能有什么影响?

回答
锁机制对性能的影响包括:

  • 锁的粒度越细,性能越高,但管理复杂度也越高。
  • 锁的持有时间越长,对性能的影响越大,可能导致死锁和性能瓶颈。
  • 合理使用锁机制可以提高并发性能,但不合理的锁设计可能导致性能下降。

MySQL锁机制的例子详解

MySQL锁机制是数据库并发控制的重要组成部分。它通过控制多个事务对数据的访问,确保数据的一致性和完整性。以下是对MySQL锁机制的详细解释,并附有示例代码。

1. 锁的类型

1.1 表锁(Table Lock)

表锁会锁住整个表,适用于全表扫描或大批量数据更新操作。

示例

CopyLOCK TABLES my_table WRITE; -- 锁定表进行写操作
-- 执行插入、更新或删除操作
UNLOCK TABLES; -- 解锁表
1.2 行锁(Row Lock)

行锁会锁住特定的行,适用于高并发环境下的细粒度控制。InnoDB存储引擎支持行锁。

示例

CopySTART TRANSACTION;
SELECT * FROM my_table WHERE id = 1 FOR UPDATE; -- 锁定特定行进行更新操作
-- 执行更新操作
COMMIT;
1.3 页锁(Page Lock)

页锁会锁住特定的页,介于表锁和行锁之间。MySQL的BDB存储引擎支持页锁,但InnoDB不支持。

2. 锁的模式

2.1 共享锁(S Lock)

共享锁允许多个事务同时读取数据,但不能修改数据。

示例

CopySTART TRANSACTION;
SELECT * FROM my_table WHERE id = 1 LOCK IN SHARE MODE; -- 加共享锁
-- 执行读取操作
COMMIT;
2.2 排他锁(X Lock)

排他锁只允许一个事务读取和修改数据,其他事务不能访问。

示例

CopySTART TRANSACTION;
SELECT * FROM my_table WHERE id = 1 FOR UPDATE; -- 加排他锁
-- 执行更新操作
COMMIT;

3. 意向锁(Intention Lock)

意向锁是一种表级锁,用于表示某个事务即将在表中的某些行上加行级锁。意向锁分为意向共享锁(IS)和意向排他锁(IX)。

4. 死锁与解决

4.1 死锁

死锁是指两个或多个事务相互等待对方持有的锁,从而导致事务无法继续执行。

4.2 解决方法
  • 死锁检测:InnoDB存储引擎会自动检测死锁,并回滚其中一个事务。
  • 超时机制:设置事务等待锁的超时时间,超过时间自动回滚。
  • 合理设计事务:尽量减少锁的持有时间,避免长时间持有锁。

5. MVCC(多版本并发控制)

MVCC通过保存数据的多个版本来实现读写并发。InnoDB存储引擎使用MVCC来实现快照读,从而避免读写冲突,提高并发性能。

6. Gap锁和Next-Key锁

6.1 Gap锁

Gap锁(间隙锁)用于锁定一个范围内的所有记录,但不包括边界值。用于防止幻读现象。

6.2 Next-Key锁

Next-Key锁是行锁和Gap锁的结合体,锁定一个范围内的所有记录,包括边界值。用于防止幻读现象。

示例代码

以下是一个综合示例,展示如何使用不同类型的锁来实现并发控制:

Copy-- 创建示例表
CREATE TABLE my_table (
    id INT PRIMARY KEY,
    value VARCHAR(255)
) ENGINE=InnoDB;

-- 插入示例数据
INSERT INTO my_table (id, value) VALUES (1, 'A'), (2, 'B'), (3, 'C');

-- 事务1:使用共享锁读取数据
START TRANSACTION;
SELECT * FROM my_table WHERE id = 1 LOCK IN SHARE MODE;
-- 执行读取操作
COMMIT;

-- 事务2:使用排他锁更新数据
START TRANSACTION;
SELECT * FROM my_table WHERE id = 1 FOR UPDATE;
-- 执行更新操作
UPDATE my_table SET value = 'Z' WHERE id = 1;
COMMIT;

-- 事务3:使用Gap锁防止幻读
START TRANSACTION;
SELECT * FROM my_table WHERE id BETWEEN 1 AND 3 FOR UPDATE;
-- 执行更新操作
COMMIT;

7. 查看锁信息

可以通过以下SQL语句查看当前的锁信息:

CopySHOW ENGINE INNODB STATUS;

或者查询information_schema中的锁相关表:

CopySELECT * FROM information_schema.INNODB_LOCKS;
SELECT * FROM information_schema.INNODB_LOCK_WAITS;

8. 性能影响与优化

锁机制对性能的影响包括:

  • 锁的粒度越细,性能越高,但管理复杂度也越高。
  • 锁的持有时间越长,对性能的影响越大,可能导致死锁和性能瓶颈。
  • 合理使用锁机制可以提高并发性能,但不合理的锁设计可能导致性能下降。

通过合理设计事务、优化SQL查询、控制锁的粒度和持有时间,可以有效地提高MySQL的并发性能

Python锁机制面试题

1. 什么是Python中的GIL(全局解释器锁)?

回答:GIL(Global Interpreter Lock,全局解释器锁)是CPython解释器中的一个互斥锁,用于保护访问Python对象的全局状态。GIL确保同一时刻只有一个线程执行Python字节码,从而避免多线程同时执行导致的竞争条件和数据不一致问题。

2. Python中的锁类型有哪些?

回答:Python中的锁类型主要包括:

  • threading.Lock:普通锁,提供基本的锁功能。
  • threading.RLock:可重入锁,允许同一线程多次获取锁。
  • threading.Condition:条件变量,提供线程间的通信机制。
  • threading.Semaphore:信号量,控制对共享资源的访问。
  • threading.Event:事件,提供线程间的信号通信。
  • threading.Barrier:屏障,阻塞一组线程直到所有线程都达到屏障位置。

3. 什么是threading.Lock

回答threading.Lock是Python中最基本的锁,用于确保同一时刻只有一个线程执行某段代码。

4. 什么是threading.RLock?与threading.Lock有何不同?

回答threading.RLock(可重入锁)允许同一线程多次获取锁,而threading.Lock在同一线程多次获取锁时会导致死锁。

5. 如何使用threading.Condition实现线程间的通信?

回答threading.Condition提供了一种线程间的通信机制,允许一个线程等待某个条件,而另一个线程通知条件满足。

6. 如何使用threading.Semaphore控制对共享资源的访问?

回答threading.Semaphore用于控制对共享资源的访问,允许指定数量的线程同时访问资源。

7. 什么是死锁?如何避免死锁?

回答:死锁是指两个或多个线程相互等待对方持有的锁,从而导致线程无法继续执行。避免死锁的方法包括:

  • 避免嵌套锁:尽量减少锁的嵌套使用。
  • 锁的顺序:确保所有线程以相同的顺序获取锁。
  • 超时机制:使用带超时的锁获取操作,避免无限等待。

8. 如何使用threading.Event实现线程间的信号通信?

回答threading.Event提供了一种线程间的信号通信机制,允许一个线程等待另一个线程发出的信号。

9. 什么是threading.Barrier?如何使用?

回答threading.Barrier用于阻塞一组线程,直到所有线程都达到屏障位置,然后所有线程同时继续执行。

10. 如何使用threading.Lock实现线程安全的计数器?

回答:可以使用threading.Lock来保护计数器的更新操作,确保线程安全。

python的锁机制 详解

Python的锁机制在多线程编程中非常重要,尤其是在需要保护共享资源的场景下。以下是对Python锁机制的详细解释:

1. GIL(全局解释器锁)

GIL(Global Interpreter Lock) 是CPython解释器中的一个互斥锁,用于保护访问Python对象的全局状态。GIL确保同一时刻只有一个线程执行Python字节码,从而避免多线程同时执行导致的竞争条件和数据不一致问题。

2. 锁的类型

2.1 threading.Lock

threading.Lock 是Python中最基本的锁,用于确保同一时刻只有一个线程执行某段代码。它提供了基本的锁和解锁功能。

  • 获取锁lock.acquire()
  • 释放锁lock.release()
import threading

lock = threading.Lock()

def critical_section():
    lock.acquire()
    try:
        # 保护的临界区代码
        print("Thread", threading.current_thread().name, "is in the critical section")
    finally:
        lock.release()
2.2 threading.RLock

threading.RLock(可重入锁)允许同一线程多次获取锁,而不会导致死锁。它在内部维护一个计数器,每次获取锁时计数器加1,释放锁时计数器减1,直到计数器为0时真正释放锁。

import threading

rlock = threading.RLock()

def recursive_function(n):
    rlock.acquire()
    try:
        if n > 0:
            print("Thread", threading.current_thread().name, "is in the recursive function with n =", n)
            recursive_function(n-1)
    finally:
        rlock.release()


 

2.3 threading.Condition

threading.Condition 提供了一种线程间的通信机制,允许一个线程等待某个条件,而另一个线程通知条件满足。它通常与LockRLock一起使用。

  • 等待条件condition.wait()
  • 通知条件condition.notify() 或 condition.notify_all()
import threading

condition = threading.Condition()
data = []

def producer():
    with condition:
        for i in range(5):
            data.append(i)
            print("Produced", i)
            condition.notify()  # 通知消费者
            condition.wait()    # 等待消费者处理

def consumer():
    with condition:
        while True:
            condition.wait()  # 等待生产者通知
            if data:
                item = data.pop(0)
                print("Consumed", item)
                condition.notify()  # 通知生产者
2.4 threading.Semaphore

threading.Semaphore 用于控制对共享资源的访问,允许指定数量的线程同时访问资源。

  • 获取信号量semaphore.acquire()
  • 释放信号量semaphore.release()
import threading
import time

semaphore = threading.Semaphore(3)  # 允许同时访问的线程数

def access_resource():
    with semaphore:
        print("Thread", threading.current_thread().name, "is accessing the resource")
        time.sleep(2)
2.5 threading.Event

threading.Event 提供了一种线程间的信号通信机制,允许一个线程等待另一个线程发出的信号。

  • 等待事件event.wait()
  • 设置事件event.set()
  • 清除事件event.clear()
import threading
import time

event = threading.Event()

def wait_for_event():
    print("Thread", threading.current_thread().name, "is waiting for the event")
    event.wait()
    print("Thread", threading.current_thread().name, "received the event")

def set_event():
    time.sleep(2)
    print("Thread", threading.current_thread().name, "is setting the event")
    event.set()
2.6 threading.Barrier

threading.Barrier 用于阻塞一组线程,直到所有线程都达到屏障位置,然后所有线程同时继续执行。

  • 等待屏障barrier.wait() import threading barrier = threading.Barrier(3) # 屏障,等待3个线程 def wait_at_barrier(): print("Thread", threading.current_thread().name, "is waiting at the barrier") barrier.wait() print("Thread", threading.current_thread().name, "passed the barrier")

3. 死锁与避免

死锁 是指两个或多个线程相互等待对方持有的锁,从而导致线程无法继续执行。避免死锁的方法包括:

  • 避免嵌套锁:尽量减少锁的嵌套使用。
  • 锁的顺序:确保所有线程以相同的顺序获取锁。
  • 超时机制:使用带超时的锁获取操作,避免无限等待。

4. 线程安全的计数器

可以使用 threading.Lock 来保护计数器的更新操作,确保线程安全。

import threading

counter = 0
lock = threading.Lock()

def increment_counter():
    global counter
    with lock:
        for _ in range(1000):
            counter += 1


 

5. 性能影响与优化

锁机制对性能的影响包括:

  • 锁的粒度:锁的粒度越细,性能越高,但管理复杂度也越高。
  • 锁的持有时间:锁的持有时间越长,对性能的影响越大,可能导致死锁和性能瓶颈。
  • 合理使用锁机制:可以提高并发性能,但不合理的锁设计可能导致性能下降。

通过合理设计事务、优化代码、控制锁的粒度和持有时间,可以有效地提高Python的并发性能。

liunx 上python编译安装或使用Anaconda同时安装运行多个不同版本的python环境 在CentOS 7上安装Python 3.7 汉化 python 两个方法均无误,已测试安装成功文章浏览阅读1.3k次,点赞58次,收藏28次。python编译安装或使用Anaconda同时安装运行多个不同版本的python环境两个方法均无误,已测试安装成功https://blog.csdn.net/qq_61414097/article/details/141402238

MySQL专栏icon-default.png?t=N7T8https://blog.csdn.net/qq_61414097/category_12758756.html

写文不易,给我点点赞和收藏吧

  • 20
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 3
    评论
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

3分云计算

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值