这篇文章主要介绍“JUC中的Lock锁与synchronized同步代码块问题怎么解决”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“JUC中的Lock锁与synchronized同步代码块问题怎么解决”文章能帮助大家解决问题。
ReentrantLock
类: 可重用锁(公平锁|非公平锁)
ReentrantReadWriteLock.ReadLock
:读锁
ReentrantReadWriteLock.WriteLock
:写锁
锁的底层有公平锁和非公平锁。其中:
公平锁 :十分公平,不能插队。
非公平锁 :十分不公平,可以插队。(默认非公平锁)
/**
* 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 加锁保证售票线程安全
*
* @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();
}
}
}
}
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适合锁大量的同步代码。
关于“JUC中的Lock锁与synchronized同步代码块问题怎么解决”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识,可以关注亿速云行业资讯频道,小编每天都会为大家更新不同的知识点。
亿速云「云服务器」,即开即用、新一代英特尔至强铂金CPU、三副本存储NVMe SSD云盘,价格低至29元/月。点击查看>>
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。