一、Lock简介
Lock 锁实现提供了比使用同步方法和语句可以获得的更广泛的锁操作。它们允许更灵活的结构,可能具有非常不同的属性,并且可能支持多个关联的条件对象。Lock 提供了比 synchronized 更多的功能。
Lock 与的 Synchronized 区别:
Lock 不是 Java 语言内置的,synchronized 是 Java 语言的关键字,因此是内置特性。Lock 是一个类,通过这个类可以实现同步访问;
Lock 和 synchronized 有一点非常大的不同,采用 synchronized 不需要用户去手动释放锁,当 synchronized 方法或者 synchronized 代码块执行完之后,系统会自动让线程释放对锁的占用;而 Lock 则必须要用户去手动释放锁,如果没有主动释放锁,就有可能导致出现死锁现象。
二、Lock中的lock()方法
lock()方法是平常使用得最多的一个方法,就是用来获取锁。如果锁已被其他线程获取,则进行等待。
采用 Lock,必须主动去释放锁,并且在发生异常时,不会自动释放锁。因此一般来说,使用 Lock 必须在 try{}catch{}块中进行,并且将释放锁的操作放在 finally 块中进行,以保证锁一定被被释放,防止死锁的发生。通常使用 Lock 来进行同步的话,是以下面这种形式去使用的:
Lock lock = ...; lock.lock(); try{ //处理任务 }catch(Exception ex){ }finally{ lock.unlock(); //释放锁 }
三、Lock的简单使用:多人卖票案例
本例实现3个人卖30张票
import java.util.concurrent.locks.ReentrantLock; /** * Lock练习,3个人卖30张票 */ //第一步:创建资源类,定义属性和操作方法 class Ticket{ //票的数量 private int number = 30; //创建可重入锁 private final ReentrantLock lock = new ReentrantLock(); //卖票方法 public void sale(){ //上锁 lock.lock(); //为了防止在解锁前出现异常而导致不能释放锁的情况,使用finally使解锁一定会执行 try{ if (number > 0){ System.out.println(Thread.currentThread().getName() + "卖出票,还剩" + --number + "张票"); } } finally { //解锁 lock.unlock(); } } } public class SaleTicket { //第二步,创建多个线程,调用资源类中的操作方法 public static void main(String[] args) { Ticket ticket = new Ticket(); new Thread(()->{ for (int i = 0; i < 40; i++) { ticket.sale(); } try { Thread.sleep(10); } catch (InterruptedException e) { e.printStackTrace(); } }, "售票员A").start(); new Thread(()->{ for (int i = 0; i < 40; i++) { ticket.sale(); } try { Thread.sleep(10); } catch (InterruptedException e) { e.printStackTrace(); } }, "售票员B").start(); new Thread(()->{ for (int i = 0; i < 40; i++) { ticket.sale(); } try { Thread.sleep(10); } catch (InterruptedException e) { e.printStackTrace(); } }, "售票员C").start(); } }
运行结果:
D:\java\jdk\jdk1.8.0_161\bin\java.exe "-javaagent:D:\IntelliJ IDEA 2021.2\lib\idea_rt.jar=49748:D:\IntelliJ IDEA 2021.2\bin" -Dfile.encoding=UTF-8 -classpath D:\Java\jdk\jdk1.8.0_161\jre\lib\charsets.jar;D:\Java\jdk\jdk1.8.0_161\jre\lib\deploy.jar;D:\Java\jdk\jdk1.8.0_161\jre\lib\ext\access-bridge-32.jar;D:\Java\jdk\jdk1.8.0_161\jre\lib\ext\cldrdata.jar;D:\Java\jdk\jdk1.8.0_161\jre\lib\ext\dnsns.jar;D:\Java\jdk\jdk1.8.0_161\jre\lib\ext\jaccess.jar;D:\Java\jdk\jdk1.8.0_161\jre\lib\ext\jfxrt.jar;D:\Java\jdk\jdk1.8.0_161\jre\lib\ext\localedata.jar;D:\Java\jdk\jdk1.8.0_161\jre\lib\ext\nashorn.jar;D:\Java\jdk\jdk1.8.0_161\jre\lib\ext\sunec.jar;D:\Java\jdk\jdk1.8.0_161\jre\lib\ext\sunjce_provider.jar;D:\Java\jdk\jdk1.8.0_161\jre\lib\ext\sunmscapi.jar;D:\Java\jdk\jdk1.8.0_161\jre\lib\ext\sunpkcs11.jar;D:\Java\jdk\jdk1.8.0_161\jre\lib\ext\zipfs.jar;D:\Java\jdk\jdk1.8.0_161\jre\lib\javaws.jar;D:\Java\jdk\jdk1.8.0_161\jre\lib\jce.jar;D:\Java\jdk\jdk1.8.0_161\jre\lib\jfr.jar;D:\Java\jdk\jdk1.8.0_161\jre\lib\jfxswt.jar;D:\Java\jdk\jdk1.8.0_161\jre\lib\jsse.jar;D:\Java\jdk\jdk1.8.0_161\jre\lib\management-agent.jar;D:\Java\jdk\jdk1.8.0_161\jre\lib\plugin.jar;D:\Java\jdk\jdk1.8.0_161\jre\lib\resources.jar;D:\Java\jdk\jdk1.8.0_161\jre\lib\rt.jar;D:\Java\juc\out\production\juc com.shang.lock.SaleTicket 售票员A卖出票,还剩29张票 售票员A卖出票,还剩28张票 售票员A卖出票,还剩27张票 售票员A卖出票,还剩26张票 售票员A卖出票,还剩25张票 售票员A卖出票,还剩24张票 售票员A卖出票,还剩23张票 售票员C卖出票,还剩22张票 售票员C卖出票,还剩21张票 售票员C卖出票,还剩20张票 售票员C卖出票,还剩19张票 售票员C卖出票,还剩18张票 售票员C卖出票,还剩17张票 售票员C卖出票,还剩16张票 售票员C卖出票,还剩15张票 售票员C卖出票,还剩14张票 售票员C卖出票,还剩13张票 售票员C卖出票,还剩12张票 售票员C卖出票,还剩11张票 售票员C卖出票,还剩10张票 售票员C卖出票,还剩9张票 售票员C卖出票,还剩8张票 售票员C卖出票,还剩7张票 售票员C卖出票,还剩6张票 售票员C卖出票,还剩5张票 售票员C卖出票,还剩4张票 售票员C卖出票,还剩3张票 售票员C卖出票,还剩2张票 售票员C卖出票,还剩1张票 售票员C卖出票,还剩0张票 Process finished with exit code 0