千家信息网

JUC中的Lock锁与synchronized同步代码块问题怎么解决

发表于:2025-01-17 作者:千家信息网编辑
千家信息网最后更新 2025年01月17日,这篇文章主要介绍"JUC中的Lock锁与synchronized同步代码块问题怎么解决"的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇"JUC中的Lock锁与sy
千家信息网最后更新 2025年01月17日JUC中的Lock锁与synchronized同步代码块问题怎么解决

这篇文章主要介绍"JUC中的Lock锁与synchronized同步代码块问题怎么解决"的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇"JUC中的Lock锁与synchronized同步代码块问题怎么解决"文章能帮助大家解决问题。

一、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的区别

  1. snchronized是内置Java关键字;Lock是一个Java类。

  2. synchronized 无法判断获取锁的状态;Lock可以判断是否获取到了锁。(boolean b = lock.tryLock();)

  3. synchronized会自动释放锁Lock必须要手动释放锁,如果不释放锁,死锁

  4. synchronized线程1获得锁阻塞时,线程2会一直等待下去;Lock锁线程1获得锁阻塞时,线程2等待足够长的时间后中断等待,去做其他的事。

  5. synchronized可重入锁:不可以中断的,非公平;Lock可重入锁:可以判断锁,非公平(可以自己设置)。

  6. lock.lockInterruptibly();方法:当两个线程同时通过该方法想获取某个锁时,假若此时线程A获取到了锁,而线程B只有在等待,那么对线程B调用threadB.interrupt()方法能够中断线程B的等待过程。

  7. synchronized适合锁少量的代码同步问题; Lock适合锁大量的同步代码。

关于"JUC中的Lock锁与synchronized同步代码块问题怎么解决"的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识,可以关注行业资讯频道,小编每天都会为大家更新不同的知识点。

0