Lock锁详解
-
从JDk5.0开始,Java提供了更强大的线程同步机制—通过显示定义同步锁对象来实现同步。同步锁使用Lock对象充当
-
java.util.concurrent.locks.Lock 接口是控制多个线程对共享资源进行访问的工具。锁提供了对共享资源的独占访问,每次只能有一个线程对Lock对象加锁,线程开始访问共享资源之前应先获得Lock对象
-
ReentrantLock 类实现了 Lock,它拥有与 synchronized 相同的并发性和内存语义,在实现线程安全的控制中,比较常用的是ReentrantLock(重进入锁) ,可以显示加锁、释放锁
Lock锁的使用格式
class A {private final ReentrantLock lock = new ReentrantLock();
public void method() {//加锁lock.lock();try {//保证线程安全的代码} finally {//解锁lock.unlock();//如果同步代码有异常,要将unlock()写入finally语句块}}
}
Lock锁的使用案例
public class TestLock {public static void main(String[] args) {TestLock2 testLock2 = new TestLock2();
new Thread(testLock2).start();new Thread(testLock2).start();new Thread(testLock2).start();}
}
class TestLock2 implements Runnable {//定义票数int ticketNums = 10;//定义lock锁private final ReentrantLock lock = new ReentrantLock();@Overridepublic void run() {while (true) {try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}
lock.lock(); //加锁try {if (ticketNums > 0) {System.out.println(ticketNums--);} else {break;}} finally {//解锁lock.unlock();}}}
}
Lock与synchronized的对比
-
Lock是显示锁(手动开启和关闭锁,别忘记关锁);synchronized 是 隐式锁,出了作用域自动释放
-
Lock只有代码块锁,synchronized 有代码块锁和方法锁
-
使用Lock锁,JVM将花费较少的时间来调度线程,性能更好。并且具有更好的扩展性(提供更多的子类)
-
优先使用顺序:
-
Lock > 同步代码块(已经进入了方法体,分配了相应资源)> 同步方法(在方法体之外)
-