鱼C论坛

 找回密码
 立即注册
查看: 143|回复: 3

请详尽地介绍一下Java中的各种锁。

[复制链接]
发表于 2025-3-18 23:32:12 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
锁机制,互斥锁等等,要求条分缕析,逻辑层次结构清晰。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

发表于 2025-3-18 23:32:44 | 显示全部楼层
在Java中,锁(Lock)是并发编程中用于控制多个线程对共享资源访问的机制。Java提供了多种锁机制,主要包括以下几类:

1. Synchronized 关键字(内置锁)  
Java最基本的锁机制是 synchronized,它是基于JVM的内置锁(Monitor Lock)。  
特点:  
      
  • 互斥性:同一时刻只有一个线程可以获取锁。  
  • 可重入性:同一线程可以多次获取同一把锁。  
  • 自动释放:线程执行完同步代码块后自动释放锁。  
  
示例:  
[code=java]  
public class SynchronizedExample {  
    private int count = 0;  

    public synchronized void increment() {  
        count++;  
    }  

    public void decrement() {  
        synchronized (this) {  
            count--;  
        }  
    }  
}  
[/code]  

2. ReentrantLock(可重入锁)  
ReentrantLock 是 java.util.concurrent.locks 包提供的显式锁,相比 synchronized,它提供了更灵活的锁机制。  
特点:  
      
  • 可重入性:同一线程可以多次获取同一把锁。  
  • 可中断性:支持 lockInterruptibly() 方法,允许线程在等待锁时被中断。  
  • 超时获取:支持 tryLock() 方法,可以尝试获取锁并设置超时时间。  
  • 公平锁/非公平锁:默认是非公平锁,可以通过构造方法指定为公平锁。  
  
示例:  
[code=java]  
import java.util.concurrent.locks.ReentrantLock;  

public class ReentrantLockExample {  
    private final ReentrantLock lock = new ReentrantLock();  
    private int count = 0;  

    public void increment() {  
        lock.lock();  
        try {  
            count++;  
        } finally {  
            lock.unlock();  
        }  
    }  
}  
[/code]  

3. ReadWriteLock(读写锁)  
ReadWriteLock 允许多个线程同时读取,但写操作是互斥的。  
特点:  
      
  • 读锁(ReadLock):多个线程可以同时获取读锁。  
  • 写锁(WriteLock):同一时刻只有一个线程可以获取写锁。  
  • 读写互斥:写锁会阻塞读锁,写锁之间也互斥。  
  
示例:  
[code=java]  
import java.util.concurrent.locks.ReentrantReadWriteLock;  

public class ReadWriteLockExample {  
    private final ReentrantReadWriteLock lock = new ReentrantReadWriteLock();  
    private int data = 0;  

    public int read() {  
        lock.readLock().lock();  
        try {  
            return data;  
        } finally {  
            lock.readLock().unlock();  
        }  
    }  

    public void write(int value) {  
        lock.writeLock().lock();  
        try {  
            data = value;  
        } finally {  
            lock.writeLock().unlock();  
        }  
    }  
}  
[/code]  

4. StampedLock(改进的读写锁)  
StampedLock 是 Java 8 引入的锁,类似于 ReadWriteLock,但提供了乐观读锁,提高了并发性能。  
特点:  
      
  • 乐观读锁:获取读锁时不会阻塞写操作,提高性能。  
  • 悲观读锁:类似于 ReadWriteLock 的读锁。  
  • 写锁:与 ReadWriteLock 的写锁类似。  
  
示例:  
[code=java]  
import java.util.concurrent.locks.StampedLock;  

public class StampedLockExample {  
    private final StampedLock lock = new StampedLock();  
    private int data = 0;  

    public int read() {  
        long stamp = lock.tryOptimisticRead();  
        int result = data;  
        if (!lock.validate(stamp)) {  
            stamp = lock.readLock();  
            try {  
                result = data;  
            } finally {  
                lock.unlockRead(stamp);  
            }  
        }  
        return result;  
    }  

    public void write(int value) {  
        long stamp = lock.writeLock();  
        try {  
            data = value;  
        } finally {  
            lock.unlockWrite(stamp);  
        }  
    }  
}  
[/code]  

5. 其他锁机制  
      
  • Semaphore(信号量):用于控制同时访问资源的线程数量。  
  • CountDownLatch(倒计时锁):用于等待多个线程完成任务。  
  • CyclicBarrier(循环屏障):让多个线程在某个点上等待,直到所有线程都到达。  
  • LockSupport(低级锁机制):提供更底层的线程阻塞和唤醒机制。  
  

总结  
Java 提供了多种锁机制,从 synchronizedReentrantLockReadWriteLockStampedLock,每种锁都有其适用场景。选择合适的锁可以提高程序的并发性能,减少线程竞争,提高吞吐量。

以上回复来自 FishC AI 小助理,如未能正确解答您的问题,请继续追问。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2025-3-18 23:34:03 | 显示全部楼层
FishC 发表于 2025-3-18 23:32
在Java中,锁(Lock)是并发编程中用于控制多个线程对共享资源访问的机制。Java提供了多种锁机制,主要包括 ...

该回答中怎么没介绍互斥锁?请补充完整。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2025-3-18 23:34:19 | 显示全部楼层
ultron110 发表于 2025-03-18 23:34
FishC 发表于 2025-3-18 23:32
在Java中,锁(Lock) ...


互斥锁(Mutex)是并发编程中最基本的锁机制之一,它保证同一时刻只有一个线程可以访问共享资源。在Java中,互斥锁的实现主要包括以下几种方式:  

### 1. **Synchronized(内置互斥锁)**  
`synchronized` 关键字本质上就是一种互斥锁,它确保同一时刻只有一个线程可以执行同步代码块。  
**特点:**  
- 互斥性:同一时刻只有一个线程可以获取锁。  
- 可重入性:同一线程可以多次获取同一把锁。  
- 自动释放:线程执行完同步代码块后自动释放锁。  

**示例:**  
startLine: 2
endLine: 10

### 2. **ReentrantLock(显式互斥锁)**  
`ReentrantLock` 是 `java.util.concurrent.locks` 包提供的显式锁,它提供了更灵活的锁机制。  
**特点:**  
- 互斥性:同一时刻只有一个线程可以获取锁。  
- 可重入性:同一线程可以多次获取同一把锁。  
- 可中断性:支持 `lockInterruptibly()` 方法,允许线程在等待锁时被中断。  
- 超时获取:支持 `tryLock()` 方法,可以尝试获取锁并设置超时时间。  

**示例:**  
startLine: 12
endLine: 20

### 3. **公平锁 vs 非公平锁**  
- **公平锁**:线程按照请求锁的顺序依次获取锁,避免线程饥饿。  
- **非公平锁**(默认):线程可能会“插队”获取锁,提高吞吐量,但可能导致某些线程长期无法获取锁。  

**示例(创建公平锁):**  
ReentrantLock lock = new ReentrantLock(true); // true 表示公平锁

### 4. **互斥锁的应用场景**  
- 保护共享资源,防止数据竞争。  
- 线程安全的计数器、队列等数据结构。  
- 需要更灵活的锁控制(如超时、可中断)时,使用 `ReentrantLock`。  

### **总结**  
互斥锁是并发编程中最常见的锁机制,Java 提供了 `synchronized` 和 `ReentrantLock` 作为主要实现方式。`synchronized` 适用于简单场景,而 `ReentrantLock` 提供了更丰富的功能,如可中断锁、超时锁和公平锁。

以上回复来自 FishC AI 小助理,如未能正确解答您的问题,请继续追问。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2025-4-4 15:38

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表