Lock (锁) 入门
介绍
- 从JDK5.0开始,Java提供了更强大的线程同步机制——通过显示定义同步锁对象来实现同步。同步锁使用Lock对象充当
- java.util.concurrent.locks.Lock接口是控制多个线程对共享资源进行访问的工具。锁提供了对共享资源的独占访问,每次只能有一个线程对Lock对象加锁,线程开始访问共享资源之前应先获得Lock对象
- ReentrantLock类实现了Lock,它拥有与synchronized相同的并发性和内存语义,在实现线程安全的控制中,比较常用的是ReentrantLock,可以显示加锁和释放锁
代码演示
import java.util.concurrent.locks.ReentrantLock;
public class LockTest {
public static void main(String[] args) {
Window w = new Window();
Thread t1 = new Thread(w);
Thread t2 = new Thread(w);
Thread t3 = new Thread(w);
t1.setName("窗口1");
t2.setName("窗口2");
t3.setName("窗口3");
t1.start();
t2.start();
t3.start();
}
}
class Window implements Runnable{
private int ticket = 100;
//构造器参数设置为true采用公平排序策略
private ReentrantLock lock = new ReentrantLock(true);
@Override
public void run() {
while (true){
try {
// 加锁
lock.lock();
if (ticket>0){
System.out.println(Thread.currentThread().getName()+":售票,票号为:"+ticket);
ticket--;
}else {
break;
}
}finally {
// 释放锁
lock.unlock();
}
}
}
}
输出:
窗口1:售票,票号为:100
窗口2:售票,票号为:99
窗口3:售票,票号为:98
窗口1:售票,票号为:97
窗口2:售票,票号为:96
窗口3:售票,票号为:95
...
窗口2:售票,票号为:3
窗口3:售票,票号为:2
窗口1:售票,票号为:1
这样也可以解决卖票的线程安全问题
synchronized和Lock的异同
相同:二者都可以解决线程安全问题
不同:synchronized机制在执行完响应的同步代码以后,自动释放同步监视器,Lock需要手动启动同步(lock()),同时结束也需要显示的释放锁(unlock())
Lock只用代码块锁,synchronized有代码块锁和方法锁
更多的内容可参考:javaguide 《谈谈 synchronized 和 ReentrantLock 的区别》https://javaguide.cn/java/concurrent/java-concurrent-questions-02.html#%E8%B0%88%E8%B0%88-synchronized-%E5%92%8C-reentrantlock-%E7%9A%84%E5%8C%BA%E5%88%AB