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

相关文章
|
23天前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
61 2
|
1月前
|
SQL 监控 数据可视化
完全开源!国内首个完全开源JAVA企业级低代码平台
JeeLowCode 是一款专为企业打造的 Java 企业级低代码开发平台,通过五大核心引擎(SQL、功能、模板、图表、切面)和四大服务体系(开发、设计、图表、模版),简化开发流程,降低技术门槛,提高研发效率。平台支持多端适配、国际化、事件绑定与动态交互等功能,广泛适用于 OA、ERP、IoT 等多种管理信息系统,帮助企业加速数字化转型。
|
12天前
|
Java 程序员
Java社招面试题:& 和 && 的区别,HR的套路险些让我翻车!
小米,29岁程序员,分享了一次面试经历,详细解析了Java中&和&&的区别及应用场景,展示了扎实的基础知识和良好的应变能力,最终成功获得Offer。
35 14
|
28天前
|
存储 算法 Java
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
本文详解自旋锁的概念、优缺点、使用场景及Java实现。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
|
28天前
|
存储 安全 Java
面试高频:Synchronized 原理,建议收藏备用 !
本文详解Synchronized原理,包括其作用、使用方式、底层实现及锁升级机制。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
面试高频:Synchronized 原理,建议收藏备用 !
|
1月前
|
存储 缓存 Oracle
Java I/O流面试之道
NIO的出现在于提高IO的速度,它相比传统的输入/输出流速度更快。NIO通过管道Channel和缓冲器Buffer来处理数据,可以把管道当成一个矿藏,缓冲器就是矿藏里的卡车。程序通过管道里的缓冲器进行数据交互,而不直接处理数据。程序要么从缓冲器获取数据,要么输入数据到缓冲器。
Java I/O流面试之道
|
27天前
|
人工智能 监控 数据可视化
Java智慧工地信息管理平台源码 智慧工地信息化解决方案SaaS源码 支持二次开发
智慧工地系统是依托物联网、互联网、AI、可视化建立的大数据管理平台,是一种全新的管理模式,能够实现劳务管理、安全施工、绿色施工的智能化和互联网化。围绕施工现场管理的人、机、料、法、环五大维度,以及施工过程管理的进度、质量、安全三大体系为基础应用,实现全面高效的工程管理需求,满足工地多角色、多视角的有效监管,实现工程建设管理的降本增效,为监管平台提供数据支撑。
35 3
|
29天前
|
存储 缓存 Java
大厂面试必看!Java基本数据类型和包装类的那些坑
本文介绍了Java中的基本数据类型和包装类,包括整数类型、浮点数类型、字符类型和布尔类型。详细讲解了每种类型的特性和应用场景,并探讨了包装类的引入原因、装箱与拆箱机制以及缓存机制。最后总结了面试中常见的相关考点,帮助读者更好地理解和应对面试中的问题。
52 4
|
1月前
|
存储 Java 程序员
Java基础的灵魂——Object类方法详解(社招面试不踩坑)
本文介绍了Java中`Object`类的几个重要方法,包括`toString`、`equals`、`hashCode`、`finalize`、`clone`、`getClass`、`notify`和`wait`。这些方法是面试中的常考点,掌握它们有助于理解Java对象的行为和实现多线程编程。作者通过具体示例和应用场景,详细解析了每个方法的作用和重写技巧,帮助读者更好地应对面试和技术开发。
86 4
|
1月前
|
Java 开发者
在Java多线程编程的世界里,Lock接口正逐渐成为高手们的首选,取代了传统的synchronized关键字
在Java多线程编程的世界里,Lock接口正逐渐成为高手们的首选,取代了传统的synchronized关键字
44 4