Java并发系列之JUC中的Lock锁与synchronized同步代码块问题
目录
- 一、Lock锁
- 二、锁的底层
- 三、案例
- 案例一:传统的synchronized实现
- 案例二:Lock锁的实现
- 四、Lock锁和synchronized的区别
本章代码:Gitee: juc.demo
一、Lock锁
ReentrantLock
类: 可重用锁(公平锁|非公平锁)ReentrantReadWriteLock.ReadLock
:读锁ReentrantReadWriteLock.WriteLock
:写锁
二、锁的底层 锁的底层有公平锁和非公平锁。其中:
- 公平锁 :十分公平,不能插队。
- 非公平锁 :十分不公平,可以插队。(默认非公平锁)
三、案例
案例一:传统的synchronized实现
/** * synchronized 同步代码块保证售票线程安全 * * @Author JUNSHI * @Create 2022-04-10 22:46 */public class SaleTicketDemo01 {public static void main(String[] args) {Ticket ticket = new Ticket(); new Thread(()->{for (int i = 0; i < 60; i++) {ticket.sale(); }},"AA").start(); },"BB").start(); new Thread(() -> {}, "CC").start(); }static class Ticket{// 50张飘票private int num = 50; // 售票 synchronized(同步代码块) 本质: 队列,锁public synchronized void sale(){if (num > 0){System.out.println(Thread.currentThread().getName()+"卖出了"+(num--)+"票,剩余:"+num); }}
案例二:Lock锁的实现
/** * Lock 加锁保证售票线程安全 * * @Author JUNSHI * @Create 2022-04-10 22:46 */public class SaleTicketDemo02 {public static void main(String[] args) {Ticket2 ticket = new Ticket2(); new Thread(()->{ for (int i = 0; i < 60; i++) ticket.sale(); },"AA").start(); new Thread(()->{ for (int i = 0; i < 60; i++) ticket.sale(); },"BB").start(); new Thread(()->{ for (int i = 0; i < 60; i++) ticket.sale(); },"CC").start(); }static class Ticket2{// 50张飘票private int num = 50; // 加锁三部曲// 1、 创建锁 => new ReentrantLock(); // 2、 加锁 =>lock.lock(); // 3、 释放锁 => lock.unlock(); public void sale(){// 可重入锁默认:非公平锁:十分不公平,可以插队。(默认非公平锁)Lock lock = new ReentrantLock(); // 加锁lock.lock(); try {// 执行业务if (num > 0){System.out.println(Thread.currentThread().getName()+"卖出了"+(num--)+"票,剩余:"+num); }} catch (Exception e) {e.printStackTrace(); } finally {// 解锁lock.unlock(); }}}}
四、Lock锁和synchronized的区别
snchronized
是内置Java关键字;Lock
是一个Java类。synchronized
无法判断获取锁的状态;Lock
可以判断是否获取到了锁。(boolean b = lock.tryLock(); )synchronized
会自动释放锁;Lock
必须要手动释放锁,如果不释放锁,死锁。synchronized
线程1获得锁阻塞时,线程2会一直等待下去;Lock锁线程1获得锁阻塞时,线程2等待足够长的时间后中断等待,去做其他的事。synchronized
可重入锁:不可以中断的,非公平;Lock可重入锁:可以判断锁,非公平(可以自己设置)。lock.lockInterruptibly()
; 方法:当两个线程同时通过该方法想获取某个锁时,假若此时线程A获取到了锁,而线程B只有在等待,那么对线程B调用threadB.interrupt()
方法能够中断线程B的等待过程。synchronized
适合锁少量的代码同步问题; Lock适合锁大量的同步代码。
推荐阅读
- Java中的私有构造函数和Singleton类
- html JavaScript脚本 – HTML教程
- 如何用Java返回多个值()
- JavaScript中的内置字符串是什么()
- 如何从JavaScript中的数组中选择一个随机元素()
- 如何在Java中查找字符串的第一个和最后一个字符
- 从列表中移除满足Java中给定谓词的元素
- JavaScript基本语法指南(注释用法实例)
- Java4Android基础学习之包与访问权限
- Java环境部署(安装和环境变量设置)