【Java面试】十七、并发篇(上)

2024-06-09 21:20
文章标签 java 面试 并发 十七

本文主要是介绍【Java面试】十七、并发篇(上),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 1、synchronized关键字的底层原理:Monitor
  • 2、synchronized相关
    • 2.1 为什么说synchronized是重量级锁
    • 2.2 synchronized锁升级之偏向锁
    • 2.3 synchronized锁升级之轻量级锁
  • 3、Java内存模型JMM
  • 4、CAS
    • 4.1 CAS流程
    • 4.2 CAS底层实现
  • 5、volatile关键字的理解
    • 5.1 可见性
    • 5.2 禁止指令重排

1、synchronized关键字的底层原理:Monitor

synchronized互斥锁,同一时刻,最多只有一个线程能持有对象锁。用javap拿到synchronized示例代码的字节码信息:

在这里插入图片描述

javap -v SyncTest.class

可以看到synchronized底层自己上锁、解锁(解锁两次是怕出现异常后导致锁不能释放,第二个解锁相当于finally里释放锁)

在这里插入图片描述

Monitor,监视器,由JVM提供,c++实现。每个对象实例都会有一个Monitor对象,Monitor的结构:

在这里插入图片描述

线程Thread1执行到synchronized代码块,如果对象的Monitor对象的Owner属性为null,则抢锁成功,后面其他线程再进来抢锁,就进入EntryList阻塞,直到Thread1执行完释放锁,EntryList里的线程又开始争抢锁(并非先来后到的排队),WaitSet即存调用了wait方法的线程

在这里插入图片描述

2、synchronized相关

2.1 为什么说synchronized是重量级锁

Java应用中的线程是用Thread对象来操作的,JVM负责维护Java线程和操作系统原生线程之间的映射关系。阻塞和唤醒一个线程,都需要CPU参与,而调用硬件资源CPU只能是内核态操作,且Java对象锁通过Monitor实现,Monitor又通过操作系统的互斥量Mutex Lock实现。

因此,加解锁、阻塞线程、唤醒线程等就涉及到了用户态和内核态的频繁切换(两个空间的一些变量的值拷贝),synchronized代码块短的话,可能切换的时间比代码执行时间还长。所以synchronized称为重量级锁。

在这里插入图片描述
鉴于此,JDK6以后,为synchronized引入轻量级锁和偏向锁的概念,避免一下就捅到重量级锁,以尽量减少用户态和内核态的切换次数。

2.2 synchronized锁升级之偏向锁

优化场景:一个锁一直被一个线程持有。如买票时发现线程t3一直在执行卖票的synchronized代码块

在这里插入图片描述
偏向锁会偏向于第一个访问锁的线程,如果在接下来的运行过程中,该锁没有被其他的线程访问,则持有偏向锁的线程将永远不需要触发同步,也即偏向锁在资源没有竞争情况下消除了同步语句。

【syncoronized偏向锁】

2.3 synchronized锁升级之轻量级锁

优化场景:两个线程近乎可以错开交替执行, 或者说是有锁竞争, 但竞争不激烈的情况,获取锁的冲突时间极端,本质就是CAS自旋锁,不要直接往重锁走。

【syncoronized轻量锁】

在这里插入图片描述

总之, synchronized的偏向锁和轻量级锁都是为了缓冲, 尽量避免走Monitor + 操作系统的互斥变量来实现加解锁, 以减少用户态和内核态的切换, 实现性能提升

3、Java内存模型JMM

定义了线程工作内存和主内存之间读写操作的规范

在这里插入图片描述

每个线程有自己的工作内存, 每块线程的工作内存之间相互隔离, 操作同一个变量时, 可通过主内存分别save和load

【JMM】

4、CAS

4.1 CAS流程

比较再交换,Compare And Swap,一种乐观锁的思想,在无锁的状态下保证线程操作数据的原子性,CAS广泛用于AQS框架、AtomicXXX类

  • 当前工作内存的值V
  • 旧的预期值A
  • 即将更新,写回主内存的值B

示意代码:

在这里插入图片描述

t1线程从主内存读到i=5,+1后准备把6写回主内存,此时,比较期望值5和内存中的实际i值,若相等,则乐观的认为自己运算的期间没有其他线程修改i,就将i写回主内存。反之,比如主内存i=6,那t1就再来一次,i=6,i+1,期望6,此时如果主内存i=6,则写回成功,反之继续自旋。

在这里插入图片描述

CAS的优势是没有加锁,线程不会陷入阻塞,效率高,反之,如果竞争激烈,频繁失败自旋重试,效率低且消耗CPU

4.2 CAS底层实现

底层通过Unsafe类来直接调用操作系统底层的CAS指令,来保证原子性,CAS对应在底层是CPU的一条原子指令cmpxchg

在这里插入图片描述

最后,CAS体现乐观锁的思想是,不担心自己操作期间别的线程修改了共享变量,如果被改了就吃点亏再重执行一次,反正我从主内存读数据、在工作内存读数据、写回主内存三步不可再分,有原子性保证,撑死多试几次才能修改数据成功

反之,synchronized则是悲观锁,自己操作时要防着其他线程改共享变量,上个锁,我改完解锁之后,别的线程才有机会

5、volatile关键字的理解

volatile修饰的变量 :

  • 保证线程间的可见性
  • 禁止进行指令重排序

5.1 可见性

volatile的可见性,即保证不同线程对某一个变量一旦完成更改,其他线程立即可见,因为会从线程的工作内存立马刷到主内存

在这里插入图片描述

执行结果:

在这里插入图片描述

结果分析:三个线程操作一个共享变量stop,线程1改了stop的值,一会儿线程2就读到了这个变化,但线程3却一直在循环,这是因为JVM的即时编译器JIT对一直执行的热点代码做了优化:

在这里插入图片描述

可加JVM参数-Xint禁用掉即时编译器,但得不偿失,可给共享变量加stop,JIT不会对volatile变量做优化且volatile变量会立即刷回主内存

static volatile boolean stop = true;

重新运行,显示循环16038233次后,读到stop变量变为true了:

在这里插入图片描述

5.2 禁止指令重排

看个案例:用@Actor注解保证方法内u的代码在同一个线程下执行

在这里插入图片描述

以上代码,并发测试下出现1 ,0 的结果,说明发生了重排序:

在这里插入图片描述
用volatile修饰变量,在读写共享变量时加入屏障,阻止其他读写操作越过屏障,以阻止指令重排序

//改为这样
int x;
volatile int y;

此时,并发测试下不再有1 ,0 的结果,屏障如下:

在这里插入图片描述
如果给x加volatile,则还是有1, 0 的结果

//改为这样
volatile int x;
int y;

因为此时的插入的屏障位置如下:

在这里插入图片描述
上面,volatile修饰x还是y,这是一个volatile的使用技巧问题:

  • 写变量让 volatile 修饰的变量的在代码最后位置
  • 读变量让 volatile 修饰的变量的在代码最开始位置

详见:volatile读写屏障

这篇关于【Java面试】十七、并发篇(上)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1046325

相关文章

Java NoClassDefFoundError运行时错误分析解决

《JavaNoClassDefFoundError运行时错误分析解决》在Java开发中,NoClassDefFoundError是一种常见的运行时错误,它通常表明Java虚拟机在尝试加载一个类时未能... 目录前言一、问题分析二、报错原因三、解决思路检查类路径配置检查依赖库检查类文件调试类加载器问题四、常见

Java注解之超越Javadoc的元数据利器详解

《Java注解之超越Javadoc的元数据利器详解》本文将深入探讨Java注解的定义、类型、内置注解、自定义注解、保留策略、实际应用场景及最佳实践,无论是初学者还是资深开发者,都能通过本文了解如何利用... 目录什么是注解?注解的类型内置注编程解自定义注解注解的保留策略实际用例最佳实践总结在 Java 编程

Java 实用工具类Spring 的 AnnotationUtils详解

《Java实用工具类Spring的AnnotationUtils详解》Spring框架提供了一个强大的注解工具类org.springframework.core.annotation.Annot... 目录前言一、AnnotationUtils 的常用方法二、常见应用场景三、与 JDK 原生注解 API 的

Java controller接口出入参时间序列化转换操作方法(两种)

《Javacontroller接口出入参时间序列化转换操作方法(两种)》:本文主要介绍Javacontroller接口出入参时间序列化转换操作方法,本文给大家列举两种简单方法,感兴趣的朋友一起看... 目录方式一、使用注解方式二、统一配置场景:在controller编写的接口,在前后端交互过程中一般都会涉及

Java中的StringBuilder之如何高效构建字符串

《Java中的StringBuilder之如何高效构建字符串》本文将深入浅出地介绍StringBuilder的使用方法、性能优势以及相关字符串处理技术,结合代码示例帮助读者更好地理解和应用,希望对大家... 目录关键点什么是 StringBuilder?为什么需要 StringBuilder?如何使用 St

使用Java将各种数据写入Excel表格的操作示例

《使用Java将各种数据写入Excel表格的操作示例》在数据处理与管理领域,Excel凭借其强大的功能和广泛的应用,成为了数据存储与展示的重要工具,在Java开发过程中,常常需要将不同类型的数据,本文... 目录前言安装免费Java库1. 写入文本、或数值到 Excel单元格2. 写入数组到 Excel表格

Java并发编程之如何优雅关闭钩子Shutdown Hook

《Java并发编程之如何优雅关闭钩子ShutdownHook》这篇文章主要为大家详细介绍了Java如何实现优雅关闭钩子ShutdownHook,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起... 目录关闭钩子简介关闭钩子应用场景数据库连接实战演示使用关闭钩子的注意事项开源框架中的关闭钩子机制1.

Maven中引入 springboot 相关依赖的方式(最新推荐)

《Maven中引入springboot相关依赖的方式(最新推荐)》:本文主要介绍Maven中引入springboot相关依赖的方式(最新推荐),本文给大家介绍的非常详细,对大家的学习或工作具有... 目录Maven中引入 springboot 相关依赖的方式1. 不使用版本管理(不推荐)2、使用版本管理(推

Java 中的 @SneakyThrows 注解使用方法(简化异常处理的利与弊)

《Java中的@SneakyThrows注解使用方法(简化异常处理的利与弊)》为了简化异常处理,Lombok提供了一个强大的注解@SneakyThrows,本文将详细介绍@SneakyThro... 目录1. @SneakyThrows 简介 1.1 什么是 Lombok?2. @SneakyThrows

在 Spring Boot 中实现异常处理最佳实践

《在SpringBoot中实现异常处理最佳实践》本文介绍如何在SpringBoot中实现异常处理,涵盖核心概念、实现方法、与先前查询的集成、性能分析、常见问题和最佳实践,感兴趣的朋友一起看看吧... 目录一、Spring Boot 异常处理的背景与核心概念1.1 为什么需要异常处理?1.2 Spring B