Java面试题之synchronized平台级锁和Lock实现的锁区别

简介: 目录一、Lock类层次结构及相关API1、Lock类层级结构2、Lock接口相关API3、关于Condition二、synchronized VS Lock1、synchronized实现的锁优缺点2、Lock实现的锁优缺点三、手撸一把简单的ReentrantLock1、ReentrantLock实现简单流程2、代码示例3、测试用例

目录

一、Lock类层次结构及相关API

1、Lock类层级结构

2、Lock接口相关API

3、关于Condition

二、synchronized VS Lock

1、synchronized实现的锁优缺点

2、Lock实现的锁优缺点

三、手撸一把简单的ReentrantLock

1、ReentrantLock实现简单流程

2、代码示例

3、测试用例

一、Lock类层次结构及相关API



1、Lock类层级结构

ReentrantLock和ReentrantReadWriteLock都是java.util.concurrent并发包下的工具类,ReentrantLock实现了Lock接口,ReentrantReadWriteLock实现了ReadWriteLock接口,而其中的ReadLock和WriteLock又实现了Lock接口。

7bf56c62e1584568abbd652fdb8253b1.png

2、Lock接口相关API

为了区别synchronized和ReentrantLock,我们先了解一下Lock接口相关的API。d22d48c26d754fd5a90b40f8d898487f.png


结论:

1、lock()最常用。

2、lockInterruptibly()方法一般更昂贵,有的impl可能没有实现lockInterruptibly,只有真的需要响应中断时才使用。

3、关于Condition

Object中的wait()、notify()、notifyAll()只能和synchronized关键字配合使用,可以唤醒一个或全部线程。Condition需要与Lock配合使用,提供多个等待集合,更精确的控制。


备注:如果说Lock代替了同步代码块或同步方法的加解锁逻辑,那么Condition则是代替了Object的等待和唤醒逻辑。

二、synchronized VS Lock



1、synchronized实现的锁优缺点

我们先说说synchronized实现的平台级锁的优点:


使用简单,语义清晰,在方法上加上synchronized关键字或者使用同步代码块即可。

由JVM提供,提供了多种优化方案,如锁粗化、锁消除、偏向锁、轻量级锁、重量级锁等,关于这些优化特性请参考:Java面试题之synchronized关键字原理以及锁相关。

锁的释放由JVM完成,不用人工干预,也降低了死锁的可能性。

再说说它的缺点:

无法实现一些锁的高级功能,如超时锁、中断锁、读写锁、共享锁、公平锁。


2、Lock实现的锁优缺点

Lock实现的锁主要弥补了synchronized的缺点,比如上面提到的锁的高级功能,如超时锁、中断锁、读写锁、共享锁、公平锁这些。


再说一下它的缺点:


需手动释放锁,新手使用不当可能造成死锁。

没有synchronized实现的锁那么多优化项。

三、手撸一把简单的ReentrantLock


1、ReentrantLock实现简单流程

先介绍一下一些关键属性,如下:

  • waiters代表锁池,说白了就是抢锁失败线程的等待队列。
  • owner代表成功获取到锁的线程。
  • count用来标记锁的可重入次数。93497eb120bc43b1adb1bba5c32044dc.png
  • 先描述下加锁流程:


如果可重入次数为0代表锁还没有被任何线程持有,这时可以通过CAS(0, count + 1)操作进行抢锁。

如果可重入次数不为0,则判断当前抢锁的线程是不是持有锁的线程,如果是则将可重入次数+1即可。

如果如上两种条件都不满足,则直接算抢锁失败。

抢锁失败的线程直接进入等待队列,并阻塞等待。

再描述下解锁流程:


如果调用解锁方法的线程不是持有锁的线程,则抛出IllegalMonitorStateException,否则第2步。

将可重入次数-1,如果可重入次数为0则代表解锁成功。

解锁成功后唤醒队列头部等待的抢锁线程。

2、代码示例

public class NicksReentrantLock implements Lock {
  // 用来标识哪个线程获取到锁
  private Thread owner;
  // 重入次数
  private AtomicInteger counter = new AtomicInteger(0);
  // 等待队列
  private BlockingQueue<Thread> waiters = new LinkedBlockingQueue<>();
  @Override
  public void lock() {
    if (!tryLock()) {
      waiters.offer(Thread.currentThread());
      // 循环解决伪唤醒问题
      while (true) {
        // 如果队列头部是当前线程说明可以抢锁
        if (Thread.currentThread() == waiters.peek()) {
          // 若抢锁成功则出队列
          if (tryLock()) {
            waiters.poll();
            return;
          }
        }
        // 若当前线程不在队列头部或者抢锁失败则挂起
        LockSupport.park();
      }
    }
  }
  @Override
  public void lockInterruptibly() throws InterruptedException {
  }
  @Override
  public boolean tryLock() {
    int ct = counter.get();
    if (ct == 0) {
      // 重入次数为0说明当前线程可以通过CAS操作获取锁
      if (counter.compareAndSet(0, ct + 1)) {
        owner = Thread.currentThread();
        return true;
      }
      return false;
    }
    // 不为0则判断获取锁的线程是否是当前线程,如果是当前线程,则将可重入次数加1
    if (owner == Thread.currentThread()) {
      counter.set(ct + 1);
      return true;
    }
    return false;
  }
  @Override
  public boolean tryLock(long time, TimeUnit unit) throws InterruptedException {
    return false;
  }
  @Override
  public void unlock() {
    // 解锁成功应该唤醒队列头部的线程
    if (tryUnlock()) {
      Optional.ofNullable(waiters.peek()).ifPresent(LockSupport::unpark);
    }
  }
  public boolean tryUnlock() {
    // 如果当前解锁的线程不是获取到锁的线程则抛异常
    if (Thread.currentThread() != owner) {
      throw new IllegalMonitorStateException();
    }
    int ct = counter.get();
    int nextCt = ct - 1;
    counter.set(nextCt);
    if (nextCt == 0) {
      owner = null;
      return true;
    }
    return false;
  }
  @Override
  public Condition newCondition() {
    return null;
  }
}


3、测试用例

public class Test {
  private static int count = 0;
  public static void main(String[] args) {
    NicksReentrantLock lock = new NicksReentrantLock();
    for (int index = 0; index < 10000; index++) {
      new Thread(() -> {
        try {
          lock.lock();
          count++;
        } finally {
          lock.unlock();
        }
      }).start();
    }
    LockSupport.parkNanos(1000 * 1000 * 1000);
    System.out.println("累加后的值为:" + count);
  }
}

备注:控制台输出为: 累加后的值为:10000

相关文章
|
29天前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
67 2
|
18天前
|
Java 程序员
Java社招面试题:& 和 && 的区别,HR的套路险些让我翻车!
小米,29岁程序员,分享了一次面试经历,详细解析了Java中&和&&的区别及应用场景,展示了扎实的基础知识和良好的应变能力,最终成功获得Offer。
48 14
|
28天前
|
存储 缓存 算法
面试官:单核 CPU 支持 Java 多线程吗?为什么?被问懵了!
本文介绍了多线程环境下的几个关键概念,包括时间片、超线程、上下文切换及其影响因素,以及线程调度的两种方式——抢占式调度和协同式调度。文章还讨论了减少上下文切换次数以提高多线程程序效率的方法,如无锁并发编程、使用CAS算法等,并提出了合理的线程数量配置策略,以平衡CPU利用率和线程切换开销。
面试官:单核 CPU 支持 Java 多线程吗?为什么?被问懵了!
|
1月前
|
存储 算法 Java
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
本文详解自旋锁的概念、优缺点、使用场景及Java实现。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
|
16天前
|
Java 关系型数据库 数据库
京东面试:聊聊Spring事务?Spring事务的10种失效场景?加入型传播和嵌套型传播有什么区别?
45岁老架构师尼恩分享了Spring事务的核心知识点,包括事务的两种管理方式(编程式和声明式)、@Transactional注解的五大属性(transactionManager、propagation、isolation、timeout、readOnly、rollbackFor)、事务的七种传播行为、事务隔离级别及其与数据库隔离级别的关系,以及Spring事务的10种失效场景。尼恩还强调了面试中如何给出高质量答案,推荐阅读《尼恩Java面试宝典PDF》以提升面试表现。更多技术资料可在公众号【技术自由圈】获取。
|
23天前
|
Java 编译器 程序员
Java面试高频题:用最优解法算出2乘以8!
本文探讨了面试中一个看似简单的数学问题——如何高效计算2×8。从直接使用乘法、位运算优化、编译器优化、加法实现到大整数场景下的处理,全面解析了不同方法的原理和适用场景,帮助读者深入理解计算效率优化的重要性。
28 6
|
存储 Java 程序员
Java面试题日积月累(数据库30道)
Java面试题日积月累(数据库30道)
70 0
|
5月前
|
SQL 安全 Java
Java面试题:什么是JDBC以及如何在Java中使用它进行数据库操作?
Java面试题:什么是JDBC以及如何在Java中使用它进行数据库操作?
55 0
|
5月前
|
druid Java 数据库连接
Java面试题:解释数据库连接池的概念及其作用,讨论常见的连接池实现。
Java面试题:解释数据库连接池的概念及其作用,讨论常见的连接池实现。
93 0
|
5月前
|
SQL Java 关系型数据库
Java面试题:描述JDBC的工作原理,包括连接数据库、执行SQL语句等步骤。
Java面试题:描述JDBC的工作原理,包括连接数据库、执行SQL语句等步骤。
72 0