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社招面试题:& 和 && 的区别,HR的套路险些让我翻车!
小米,29岁程序员,分享了一次面试经历,详细解析了Java中&和&&的区别及应用场景,展示了扎实的基础知识和良好的应变能力,最终成功获得Offer。
68 14
|
1月前
|
存储 缓存 算法
面试官:单核 CPU 支持 Java 多线程吗?为什么?被问懵了!
本文介绍了多线程环境下的几个关键概念,包括时间片、超线程、上下文切换及其影响因素,以及线程调度的两种方式——抢占式调度和协同式调度。文章还讨论了减少上下文切换次数以提高多线程程序效率的方法,如无锁并发编程、使用CAS算法等,并提出了合理的线程数量配置策略,以平衡CPU利用率和线程切换开销。
面试官:单核 CPU 支持 Java 多线程吗?为什么?被问懵了!
|
1月前
|
Java 编译器 程序员
Java面试高频题:用最优解法算出2乘以8!
本文探讨了面试中一个看似简单的数学问题——如何高效计算2×8。从直接使用乘法、位运算优化、编译器优化、加法实现到大整数场景下的处理,全面解析了不同方法的原理和适用场景,帮助读者深入理解计算效率优化的重要性。
36 6
|
1月前
|
SQL 安全 Java
JavaSecLab 一款综合Java漏洞平台
JavaSecLab是一款综合型Java漏洞学习平台,涵盖多种漏洞场景,提供漏洞代码、修复示例、安全编码规范及友好UI。适用于安全服务、甲方安全培训、安全研究等领域,助于理解漏洞原理与修复方法。支持跨站脚本、SQL注入等多种漏洞类型……
|
1月前
|
监控 前端开发 Java
【技术开发】接口管理平台要用什么技术栈?推荐:Java+Vue3+Docker+MySQL
该文档介绍了基于Java后端和Vue3前端构建的管理系统的技术栈及功能模块,涵盖管理后台的访问、登录、首页概览、API接口管理、接口权限设置、接口监控、计费管理、账号管理、应用管理、数据库配置、站点配置及管理员个人设置等内容,并提供了访问地址及操作指南。
|
25天前
|
人工智能 移动开发 安全
家政上门系统用户端、阿姨端源码,java家政管理平台源码
家政上门系统基于互联网技术,整合大数据分析、AI算法和现代通信技术,提供便捷高效的家政服务。涵盖保洁、月嫂、烹饪等多元化服务,支持多终端访问,具备智能匹配、在线支付、订单管理等功能,确保服务透明、安全,适用于家庭生活的各种需求场景,推动家政市场规范化发展。
|
6月前
|
安全 Java 程序员
Java并发编程中的锁机制与优化策略
【6月更文挑战第17天】在Java并发编程的世界中,锁是维护数据一致性和线程安全的关键。本文将深入探讨Java中的锁机制,包括内置锁、显式锁以及读写锁的原理和使用场景。我们将通过实际案例分析锁的优化策略,如减少锁粒度、使用并发容器以及避免死锁的技巧,旨在帮助开发者提升多线程程序的性能和可靠性。
|
5月前
|
存储 缓存 Java
Java面试题:解释Java中的内存屏障的作用,解释Java中的线程局部变量(ThreadLocal)的作用和使用场景,解释Java中的锁优化,并讨论乐观锁和悲观锁的区别
Java面试题:解释Java中的内存屏障的作用,解释Java中的线程局部变量(ThreadLocal)的作用和使用场景,解释Java中的锁优化,并讨论乐观锁和悲观锁的区别
60 0
|
7月前
|
安全 Java 编译器
Java并发编程中的锁优化策略
【5月更文挑战第30天】 在多线程环境下,确保数据的一致性和程序的正确性是至关重要的。Java提供了多种锁机制来管理并发,但不当使用可能导致性能瓶颈或死锁。本文将深入探讨Java中锁的优化策略,包括锁粗化、锁消除、锁降级以及读写锁的使用,以提升并发程序的性能和响应能力。通过实例分析,我们将了解如何在不同场景下选择和应用这些策略,从而在保证线程安全的同时,最小化锁带来的开销。
|
7月前
|
安全 Java 开发者
Java并发编程中的锁优化策略
【5月更文挑战第30天】 在Java并发编程领域,锁机制是实现线程同步的关键手段之一。随着JDK版本的发展,Java虚拟机(JVM)为提高性能和降低延迟,引入了多种锁优化技术。本文将深入探讨Java锁的优化策略,包括偏向锁、轻量级锁以及自旋锁等,旨在帮助开发者更好地理解和应用这些高级特性以提升应用程序的性能。