【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面试】十七、并发篇(上)
文章目录 1、synchronized关键字的底层原理:Monitor2、synchronized相关2.1 为什么说synchronized是重量级锁2.2 synchronized锁升级之偏向锁2.3 synchronized锁升级之轻量级锁 3、Java内存模型JMM4、CAS4.1 CAS流程4.2 CAS底层实现 5、volatile关键字的理解5.1 可见…...
七天进阶elasticsearch[one]
elasticSearch 概述 Elasticsearch是一个近实时的搜索平台。这意味着,从索引一个文档直到这个文档能够被搜索到有一个很小的延迟(通常是一秒) 集群 一个集群就是由一个或多个节点组织在一起, 它们共同持有你全部的数据&#x…...
全新抖音快手小红书视频解析去水印系统网站源码
这个系统支持几十种平台,包括抖音、快手小红书以及其他热门社交媒体平台。它可以帮助轻松地下载这些平台上的任何视频,并去除其中的水印,让你可以自由地保存和分享这些视频。 使用方法: 上传压缩包解压,网站信息在inc…...
图解 Python 编程(12) | 文件和编码方式
🌞欢迎来到Python 的世界 🌈博客主页:卿云阁 💌欢迎关注🎉点赞👍收藏⭐️留言📝 🌟本文由卿云阁原创! 📆首发时间:🌹2024年6月9日&am…...
CSDN个人主页动态地图(前端/后端)
前段时间有人问我,关于CSDN个人主页动态地图的实现,我来解答一下。 这里要注意一下:使用CSDN的API需要遵循其开发者协议和使用规范,确保你的使用方式符合相关规定 前端部分: 创建一个HTML页面作为个人主页。在页面上放…...
GUI编程-01
组件 窗口 弹窗 面板 文本框 列表框 按钮 图片 监听事件 鼠标 键盘事件 破解工具 Java提供了丰富的图形用户界面(Graphics User Interface,GUI)的类库,基于这些类库可以编写窗口程序。 Java关于图形界面的类库主要放在…...
Linux网络-自定义协议、序列化和反序列化、网络计算服务器的实现和Windows端客户端
文章目录 前言一、自定义协议传结构体对象 序列化和反序列化什么是序列化?反序列化 二、计算器服务端(线程池版本)1.main.cc2.Socket.hpp3.protocol.hpp4.Calculator.hpp5.serverCal.hpp6.threadPool.hpp7.Task.hpp8. log.hpp 客户端Windows客…...
Hive知识体系保姆级教程
一. Hive概览 1.1 hive的简介 Hive是基于Hadoop的一个数据仓库工具,可以将结构化的数据文件映射为一张数据库表,并提供类SQL查询功能。 其本质是将SQL转换为MapReduce/Spark的任务进行运算,底层由HDFS来提供数据的存储,说白了h…...
三大网络简介
一、三大网是哪三大网 三大网络为电话网、电视广播网、互联网,如果这三大网络使用都是“ip分组交换”技术的话,他们将会被融合成为一个网络, 但是由于历史原因,他们使用了不同的通信技术,三大网各自拥有相当的独立性&a…...
Element-UI全面入门与实战技巧
本文详细介绍了Element-UI的安装、配置、组件使用、布局技巧、交互设计、表单处理、主题定制等内容,旨在帮助开发者快速掌握Element-UI,并能在实际项目中灵活应用。 文章目录 一、Element-UI概述与安装1.1 Element-UI简介1.2 环境搭建1.3 安装Element-UI…...
第103天: 权限提升-Linux 系统辅助项目脏牛Dirty内核漏洞SUIDGUID
项目下载地址 综合类探针: https://github.com/liamg/traitor 自动化提权: https://github.com/AlessandroZ/BeRoot 信息收集: https://github.com/rebootuser/LinEnum https://github.com/sleventyeleven/linuxprivchecker 漏洞探针…...
如何用群晖当异地组网服务器?
在当今信息化时代,远程通信成为了企业和个人之间不可或缺的一部分。特别是对于跨地区的通信需求,一个可靠的异地组网服务器是必不可少的。而群晖(Synology)作为一款功能强大的网络存储设备,可以被用作办公室或家庭的异…...
文件怎么去重?5个技巧,教你删除重复文件!
一般来说,在处理大量文件时,你可能会遇到重复的类似文件。这些文件占据了电脑上不必要的磁盘空间,导致系统性能下降。而这些文件可以是不同类型的,如照片、视频、音频、存档、文档等。正因如此,您需要通过文件去重来删…...
标准发布实施 | 《村镇污水处理一体化集成装备技术规范》
根据《中华人民共和国标准化法》以及国家标准化管理委员会、民政部联合制定的《团体标准管理规定》,依据全国团体标准信息平台和《中华环保联合会团体标准管理办法(试行)》,全国团体标准《村镇污水处理一体化集成装备技术指南》&a…...
人工智能--教育领域的运用
文章目录 🐋引言 🐋个性化学习 🦈体现: 🦈技术解析: 🐋智能辅导与虚拟助手 🦈体现: 🦈技术解析: 🐋自动评分与评估 …...
【设计模式深度剖析】【3】【行为型】【职责链模式】| 以购物中心客户服务流程为例加深理解
👈️上一篇:命令模式 | 下一篇:策略模式👉️ 设计模式-专栏👈️ 文章目录 职责链模式定义英文原话直译如何理解呢? 职责链模式的角色1. Handler(抽象处理者)2. ConcreteHandler(具体处理者…...
评价GPT-4的方案
评价GPT-4的方案 引言: 随着人工智能技术的不断发展,自然语言处理领域取得了显著的突破。其中,GPT-4作为最新的大型语言模型之一,备受关注。本方案旨在对GPT-4进行全面评价,包括其技术特点、性能表现、应用场景以及潜在的影响等方面。 一、技术特点 1. 模型规模和参数数…...
LeetCode | 1624.两个相同字符之间的最长子字符串
这道题拿到手想法就是去双重遍历暴力解,对于每个字符,从后往前遍历字符串,找到从后往前一直到本次遍历的这个字符串这段子串中和这个字符串相同的字符位置,然后得到子字符串的长度,和ans存储的值做一个比较,…...
【CS.AI】GPT-4o:重新定义人工智能的新标杆
文章目录 1 序言2 GPT-4o的技术亮点3 GPT-4o与前代版本的对比3.1 热门AI模型对比表格GPT-3.5GPT-4GPT-4oBERTT5 3.2 其他 4 个人体验与感受5 结论 1 序言 嘿,大家好!今天要聊聊一个超级酷的AI新突破——GPT-4o!最近,OpenAI发布了…...
野火FPGA跟练(四)——串口RS232、亚稳态
目录 简介接口与引脚通信协议亚稳态RS232接收模块模块框图时序波形RTL 代码易错点Testbench 代码仿真 RS232发送模块模块框图时序波形RTL 代码Testbench 代码仿真 简介 UART:Universal Asynchronous Receiver/Transmitter,异步串行通信接口。发送数据时…...
在微服务架构中利用Taotoken统一管理多模型API调用与成本
在微服务架构中利用Taotoken统一管理多模型API调用与成本 1. 微服务架构中的AI能力集成挑战 现代微服务架构通常需要集成多种AI模型能力,例如自然语言处理、代码生成或数据分析。传统直接对接各厂商API的方式会面临几个典型问题:每个服务需要单独管理A…...
从零开始使用 Taotoken 在本地运行 AI 辅助编程工作流
从零开始使用 Taotoken 在本地运行 AI 辅助编程工作流 1. 准备工作 在开始配置之前,请确保您已经完成以下准备工作。首先,访问 Taotoken 平台并注册账号,在控制台中创建 API Key。其次,在模型广场查看并记录您希望使用的模型 ID…...
从ChatGPT到CowAgent:AI Agent框架的部署与实战指南
1. 项目概述:从聊天机器人到超级AI助理的蜕变 如果你和我一样,在2023年左右就开始折腾各种大模型应用,那你大概率听说过或者用过 chatgpt-on-wechat 这个项目。它曾经是很多开发者将GPT能力接入微信、飞书等国内IM平台的首选方案。但如果你…...
初创公司如何以最小成本起步验证ai产品想法
初创公司如何以最小成本起步验证AI产品想法 1. 验证阶段的成本挑战与应对思路 对于资源有限的初创团队而言,验证AI产品原型的核心挑战往往集中在三个方面:模型选型的不确定性、接入多个模型的复杂性以及早期成本不可控的风险。传统方式需要为每个候选模…...
经验分享:我的AI产品经理自学之路,靠AIPM少走90%弯路
当下AI产品经理已成风口,但零基础没人带、知识太零散、求职无背书,是大多数人自学的三大痛点。我从纯小白到系统入门,全程靠自学一张证书打通路径,今天把真实可复制的路线写出来,不吹不黑,适合所有想入行、…...
Win10/11学汇编不求人:手把手教你用DosBox+Debug搭建复古调试环境(附debug命令速查表)
Win10/11学汇编不求人:手把手教你用DosBoxDebug搭建复古调试环境 在64位Windows系统普及的今天,学习汇编语言时最令人头疼的莫过于找不到合适的调试工具。许多高校计算机组成原理课程仍在使用经典的debug.exe,但这个工具早已从现代Windows系统…...
微信聊天记录永久保存指南:开源工具WeChatExporter让回忆不再丢失
微信聊天记录永久保存指南:开源工具WeChatExporter让回忆不再丢失 【免费下载链接】WeChatExporter 一个可以快速导出、查看你的微信聊天记录的工具 项目地址: https://gitcode.com/gh_mirrors/wec/WeChatExporter 你是否曾因手机丢失、系统升级或误操作而丢…...
利用快马平台快速构建代码审查关系图可视化原型
最近在团队协作开发时,经常遇到代码审查效率不高的问题。大家修改的文件相互关联,但仅通过文字描述很难直观理解变更之间的影响关系。于是我想尝试做一个可视化工具,把代码审查中的依赖关系用图形展示出来。在InsCode(快马)平台上,…...
3分钟解锁Windows触控板三指拖拽:告别繁琐操作,提升效率300%
3分钟解锁Windows触控板三指拖拽:告别繁琐操作,提升效率300% 【免费下载链接】ThreeFingersDragOnWindows Enables macOS-style three-finger dragging functionality on Windows Precision touchpads. 项目地址: https://gitcode.com/gh_mirrors/th/T…...
别再只会用echo $PATH了!Linux环境变量获取的四种C语言实现方式(附完整代码)
深入Linux环境变量:C语言程序员的四种高效获取方式 在Linux系统编程中,环境变量是进程运行环境的重要组成部分。对于C语言开发者来说,掌握环境变量的获取方式不仅是基础技能,更是编写健壮系统程序的关键。本文将深入探讨四种C语言…...
