面试题16解析-深挖锁(下)

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
简介: 本文阅读大概需要15分钟。接着面试题16未讲完的部分,咱们继续来聊聊锁这个话题


一重入锁(ReentrantLock)


Java SE5以后,Java并发包基于Lock接口,实现了高性能的支持重入的锁ReentrantLock。重入这里指的是在某线程已经获取锁之后,该线程可以再次获取锁,进入同步代码块。这里需要强调一下重入的概念中所指的线程是已经获得锁的的线程,这与线程安全不冲突,因为只有一个线程可以获取锁,也就是说始终都是同一个线程(获取锁的线程)在运行同步代码,相当于单线程运行,当然是线程安全的。synchronized关键字也是支持重入的,例如synchronized可以在递归调用中使用。


以下列出了synchronized与ReentrantLock的不同之处:

  1. ReentrantLock提供了显式加解锁操作。提供了lock(),unlock()方法进行加解锁的操作,而synchronized是隐式进行加锁与解锁操作(依赖于编译器将其编译为moniterenter与moniterexit)。
  2. 对锁的等待可以中断,在持有锁的线程长时间不释放锁时,等待锁的线程可以选择放弃等待,这样就避免了synchronized可能带来的死锁问题。ReentrantLock.tryLock()可以设置等待时间。
  3. ReentrantLock提供了公平锁与非公平锁,而synchronized的实现是非公平锁。


ReentrantLock相对于synchronized来说一般用于,加锁与解锁操作需要分离的使用场景,例如加解锁不再一个函数里(synchronized无法用括号包围),相对来说ReentrantLock提供了更高的灵活性,但是使用时一定不要忘了释放锁。


二读写锁(ReentrantReadWriteLock)


加锁是我们为了保证共享资源在多线程操作下不会出现脏读、脏存现象,因此需要对共享资源进行了加锁保护。但是在大部分系统中,变量的读操作远远多于变量的写操作,大多情况下变量的值都是不变的,如果我们可以去除锁,并保证变量的改变能对所有线程可以及时可见,那么也不会存在脏读现象,这样就可以大幅度的提升系统的并发性能。读写锁正是基于上述需求产生的,相对与ReentrantLock它对变量的读写操作进行了区别对待,遵循以下特性:

  1. 在没有写操作线程获取写锁的情况下,所有读操作都可以获取读锁。
  2. 在有写线程获取写锁的情况下,读操作等待写线程释放锁后,才可以获取读锁。
  3. 在有读线程获取读锁的情况下,写线程会等待所有读线程释放锁后,才可以获取写锁,并且与此同时,所有的读锁也不可获取。

综上所述,就是保证读写不会同时发生。下面我们通过开发一个线程安全的读写分离HashMap来看看读写锁的具体使用:

public class ReentrantReadWriteLockHashMap {
    private final Map<String, Object> hashMap = new HashMap<String, Object>();
    private final ReentrantReadWriteLock rwLock = new ReentrantReadWriteLock();
    private final Lock readLock = rwLock.readLock();
    private final Lock writeLock = rwLock.writeLock();
    public final void put(String key, Object value){
        //上写锁,不允许其他线程读也不允许写
        writeLock.lock();
        hashMap.put(key, value);
        writeLock.unlock();
    }
    public final Object get( String key ){
        //上读锁,其他线程只能读不能写
        readLock.lock();
        Object value =  hashMap.get(key);
        readLock.unlock();
        return value;
    }
}

通过示例代码,可以看到读写锁的使用还是很简单的,关键要理解读写锁的运行机制,读写分离,区别对待。

到此,对这个题目讲解就结束了,你应该对Java中的锁有所了解了吧。

相关文章
|
2月前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
87 2
|
1月前
|
Java 程序员
面试官的加分题:super关键字全解析,轻松应对!
小米,29岁程序员,通过一个关于Animal和Dog类的故事,详细解析了Java中super关键字的多种用法,包括调用父类构造方法、访问父类成员变量及调用父类方法,帮助读者更好地理解和应用super,应对面试挑战。
41 3
|
2月前
|
存储 网络协议 安全
30 道初级网络工程师面试题,涵盖 OSI 模型、TCP/IP 协议栈、IP 地址、子网掩码、VLAN、STP、DHCP、DNS、防火墙、NAT、VPN 等基础知识和技术,帮助小白们充分准备面试,顺利踏入职场
本文精选了 30 道初级网络工程师面试题,涵盖 OSI 模型、TCP/IP 协议栈、IP 地址、子网掩码、VLAN、STP、DHCP、DNS、防火墙、NAT、VPN 等基础知识和技术,帮助小白们充分准备面试,顺利踏入职场。
103 2
|
2月前
|
存储 NoSQL MongoDB
MongoDB面试专题33道解析
大家好,我是 V 哥。今天为大家整理了 MongoDB 面试题,涵盖 NoSQL 数据库基础、MongoDB 的核心概念、集群与分片、备份恢复、性能优化等内容。这些题目和解答不仅适合面试准备,也是日常工作中深入理解 MongoDB 的宝贵资料。希望对大家有所帮助!
|
2月前
|
缓存 前端开发 JavaScript
"面试通关秘籍:深度解析浏览器面试必考问题,从重绘回流到事件委托,让你一举拿下前端 Offer!"
【10月更文挑战第23天】在前端开发面试中,浏览器相关知识是必考内容。本文总结了四个常见问题:浏览器渲染机制、重绘与回流、性能优化及事件委托。通过具体示例和对比分析,帮助求职者更好地理解和准备面试。掌握这些知识点,有助于提升面试表现和实际工作能力。
69 1
|
4月前
|
设计模式 Java 关系型数据库
【Java笔记+踩坑汇总】Java基础+JavaWeb+SSM+SpringBoot+SpringCloud+瑞吉外卖/谷粒商城/学成在线+设计模式+面试题汇总+性能调优/架构设计+源码解析
本文是“Java学习路线”专栏的导航文章,目标是为Java初学者和初中高级工程师提供一套完整的Java学习路线。
503 37
|
3月前
|
NoSQL Java API
美团面试:Redis锁如何续期?Redis锁超时,任务没完怎么办?
在40岁老架构师尼恩的读者交流群中,近期有小伙伴在面试一线互联网企业时遇到了关于Redis分布式锁过期及自动续期的问题。尼恩对此进行了系统化的梳理,介绍了两种核心解决方案:一是通过增加版本号实现乐观锁,二是利用watch dog自动续期机制。后者通过后台线程定期检查锁的状态并在必要时延长锁的过期时间,确保锁不会因超时而意外释放。尼恩还分享了详细的代码实现和原理分析,帮助读者深入理解并掌握这些技术点,以便在面试中自信应对相关问题。更多技术细节和面试准备资料可在尼恩的技术文章和《尼恩Java面试宝典》中获取。
美团面试:Redis锁如何续期?Redis锁超时,任务没完怎么办?
|
4月前
|
消息中间件 安全 前端开发
面试官:单核服务器可以不加锁吗?
面试官:单核服务器可以不加锁吗?
56 4
面试官:单核服务器可以不加锁吗?
|
3月前
|
存储 安全 Java
JVM锁的膨胀过程与锁内存变化解析
在Java虚拟机(JVM)中,锁机制是确保多线程环境下数据一致性和线程安全的重要手段。随着线程对共享资源的竞争程度不同,JVM中的锁会经历从低级到高级的膨胀过程,以适应不同的并发场景。本文将深入探讨JVM锁的膨胀过程,以及锁在内存中的变化。
58 1
|
4月前
|
缓存 Android开发 开发者
Android RecycleView 深度解析与面试题梳理
本文详细介绍了Android开发中高效且功能强大的`RecyclerView`,包括其架构概览、工作流程及滑动优化机制,并解析了常见的面试题。通过理解`RecyclerView`的核心组件及其优化技巧,帮助开发者提升应用性能并应对技术面试。
114 8

推荐镜像

更多