线程安全、线程的同步(5)Lock锁方式解决线程安全问题
一、 Lock(锁)说明
- 从JDK 5.0开始,Java提供了更强大的线程同步机制——通过显式定义同步锁对象来实现同步。同步锁使用Lock对象充当。
- **java.util.concurrent.locks.Lock接口是控制多个线程对共享资源进行访问的工具。**锁提供了对共享资源的独占访问,每次只能有一个线程对Lock对象加锁,线程开始访问共享资源之前应先获得Lock对象。
- ReentrantLock 类实现了 Lock ,它拥有与 synchronized 相同的并发性和内存语义,在实现线程安全的控制中,比较常用的是ReentrantLock,可以显式加锁、释放锁。
二、优先使用顺序:
Lock----》同步代码块(已经进入了方法体,分配了相应资源)----》同步方法(在方法体之外)
三、面试题:synchronized 与 Lock的异同?
- 相同:二者都可以解决线程安全问题
- 不同:synchronized机制在执行完相应的同步代码以后,自动的释放同步监视器;Lock需要手动的启动同步(lock()),同时结束同步也需要手动的实现(unlock())
代码:
package java1;
import java.util.concurrent.locks.ReentrantLock;
class Window implements Runnable {
private int ticket = 30;
//1.实例化ReentrantLock
private ReentrantLock lock = new ReentrantLock();
@Override
public void run() {
while (true) {
try {
//2.调用锁定方法lock()
lock.lock();
if (ticket > 0) {
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + ":售票,票号为:" + ticket);
ticket--;
} else {
break;
}
} finally {
//3.调用解锁方法:unlock()
lock.unlock();
}
}
}
}
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();
}
}
输出:
窗口1:售票,票号为:30
窗口1:售票,票号为:29
窗口1:售票,票号为:28
窗口1:售票,票号为:27
窗口1:售票,票号为:26
窗口1:售票,票号为:25
窗口1:售票,票号为:24
窗口1:售票,票号为:23
窗口1:售票,票号为:22
窗口1:售票,票号为:21
窗口1:售票,票号为:20
窗口2:售票,票号为:19
窗口2:售票,票号为:18
窗口2:售票,票号为:17
窗口2:售票,票号为:16
窗口2:售票,票号为:15
窗口2:售票,票号为:14
窗口2:售票,票号为:13
窗口2:售票,票号为:12
窗口2:售票,票号为:11
窗口2:售票,票号为:10
窗口2:售票,票号为:9
窗口2:售票,票号为:8
窗口2:售票,票号为:7
窗口2:售票,票号为:6
窗口2:售票,票号为:5
窗口2:售票,票号为:4
窗口2:售票,票号为:3
窗口2:售票,票号为:2
窗口2:售票,票号为:1