作者 | SoWhat1412

来源 | SoWhat1412(id:sowhat9094)

头图 | 东方IC

今天闲来无事跟同事大帆闲聊:

SoWhat:麦叔听说你偷偷面阿里啦,面的咋样?

大帆:一面挺简单的,主要问了一些基本的数据结构跟算法,还问了下  HashMap的十大常见基本问题。我都答案上来了,还问了我JDK7环,幸亏你那个 HashMap环 绘制的牛逼,我答的不错就让我准备二面了。

SoWhat:二面类?

大帆:二面问了我一些JVM的问题,问我对于JVM内存模型的理解,还有GC的常见理解,最终还问了我下类加载机制,我看你之前水过这个 JVM系列,就依葫芦画瓢答上来了,让我准备三面。

SoWhat:大帆这波可以啊,三面问的啥啊?

大帆:三面问了我一些CAS、Lock、AQS跟 ConcurrentHashMap 的底层实现什么的,还问了我下 线程池 的七大参数跟四大拒绝策略,以及使用注意事项。我看你水过 并发编程系列,也就答上来了。

Sowhat:厉害啊这是要过的节奏阿!

大帆:过个锤子,三面的这个总监最后竟然问了我下我对volatile的底层原理。你妹的你么水,我就答了一些基本的可见性跟弱原子性,然后我感觉面试官不太满意啊!

Sowhat:额好吧,那我抓紧再水文写下个关于volatile的使用。

使用

volatile变量自身具有下列特性相信大家都知道:

可见性。对一个volatile变量的读,总是能看到(任意线程)对这个volatile变量最后的写入。

原子性:对任意单个volatile变量的读/写具有原子性,但类似于volatile++这种复合操作不具有原子性。

其中第二点可以理解为把对 volatile 变量的单个读/写,看成是使用同一个锁对这些单个读/写操作做了同步,就跟下面的SoWhat跟SynSoWhat功能类似哦。

class SoWhat{volatile int i = 0; // volatile修饰的变量public int getI(){return i;// 单个volatile变量的读}public  void setI(int j){this.i = j; // 单个volatile 变量的写}public void inc(){i++;//复合多个volatile 变量}
}
class SynSoWhat{int i = 0;public synchronized int getI(){return i;}public  synchronized void setI(int j){this.i = j;}public void inc(){ // 普通方法调用int tmp = getI(); // 调用已同步方法tmp = tmp + 1;//普通写方法setI(tmp);// 调用已同步方法}
}

写理解

volatile写的内存语义如下:

当写一个volatile变量时,JMM会把该线程对应的本地中的共享变量值刷新到主内存。

public class VolaSemanteme {int a = 0;volatile boolean flag = false; // 这是重点哦public void init() {a = 1; flag = true; //.......}public void use() {if (flag) { int i = a * a; }//.......}
}线程A调用init方法,线程B调用use方法。

线程A调用init方法,线程B调用use方法。

读理解

volatile读的内存语义如下:

当读一个volatile变量时,JMM会把该线程对应的本地内存置为无效。线程接下来将从主内存中读取共享变量。

public class VolaSemanteme {int a = 0;volatile boolean flag = false; // 这是重点哦public void init() {a = 1; flag = true; //.......}public void use() {if (flag) { int i = a * a; }//.......}
}

流程图大致是这样的:

volatile 指令重排

volatile 变量的内存可见性是基于内存屏障(Memory Barrier)实现。关于内存屏障的具体讲解以前写过不再重复,JMM装逼于无形这里说过。总结来说就是JMM内部会有指令重排,并且会有af-if-serial跟happen-before的理念来保证指令重拍的正确性。内存屏障就是基于4个汇编级别的关键字来禁止指令重排的,其中volatile的重拍规则如下:

第一个为读操作时,第二个任何操作不可重排序到第一个前面。

第二个为写操作时,第一个任何操作不可重排序到第二个后面。

第一个为写操作时,第二个的读写操作也不运行重排序。

volatile写底层实现

JMM对volatile的内存屏障插入策略

在每个volatile写操作的前面插入一个StoreStore屏障。在每个volatile写操作的后面插入一个StoreLoad屏障。

volatile 读底层

JMM对volatile的内存屏障插入策略

在每个volatile读操作的后面插入一个LoadLoad屏障。在每个volatile读操作的后面插入一个LoadStore屏障。

其中重点说下volatile读后面为什么跟了个LoadLoad。加入我有如下代码 AB两个线程执行,B线程的flag获取下面的读被提前了。

volatile的实现原理

有volatile变量修饰的共享变量进行写操作的时候会使用CPU提供的Lock前缀指令。在CPU级别的功能如下:

将当前处理器缓存行的数据写回到「系统内存」

这个写回内存的操作会告知在其他CPU你们拿到的变量是无效的下一次使用时候要重新共享内存拿。

我们可以通过jitwatch对简单的代码进行详细的反汇编看一下。

package com.sowhat.demo;public class VolaSemanteme {int unvloatileVal = 0;volatile boolean flag = false;public void init() {unvloatileVal = 1;flag = true; // 第九行哦}public void use() {if (flag) {int LocalA = unvloatileVal;if (LocalA == 0) {throw new RuntimeException("error");}}}public static void main(String[] args) {VolaSemanteme volaSemanteme = new VolaSemanteme();volaSemanteme.init();volaSemanteme.use();}
}

对普通变量的赋值操作:

对volatile变量的赋值操作:

可以对比得出,volatile 修饰的变量确实会多一个 lock addl $0x0,(%rsp) 指令。

0x0000000114ce95cb: lock addl $0x0,(%rsp)  ;*putfield flag; - com.sowhat.demo.VolaSemanteme::init@7 (line 9)更多精彩推荐
☞倪光南、求伯君“出山”:爱解 Bug、无惧“35岁魔咒”、编码之路痛并快乐!
☞我坦白!我是第五位飞上太空的程序员游客
☞腾讯回应发布虚假广告被罚20万;苹果客服回应iPhone 12屏幕发绿;Chrome 87 正式版发布|极客头条
☞赠书 | 图像分类问题建模方案探索实践☞大神们都是如何在时间序列中进行特征提取的?看完就懂了!
☞Value DeFi遭黑客攻击始末,闪电贷这次又带走了700万美元
点分享点点赞点在看

三面阿里竟然败在了 volatile 关键字上相关推荐

  1. Java实战应用50篇(一)-Java并发编程:volatile关键字解析

    前言 volatile这个关键字可能很多朋友都听说过,或许也都用过.在Java 5之前,它是一个备受争议的关键字,因为在程序中使用它往往会导致出人意料的结果.在Java 5之后,volatile关键字 ...

  2. java中volatile_java中volatile关键字的含义

    在java线程并发处理中,有一个关键字volatile的使用目前存在很大的混淆,以为使用这个关键字,在进行多线程并发处理的时候就可以万事大吉. Java语言是支持多线程的,为了解决线程并发的问题,在语 ...

  3. Java并发编程--volatile关键字解析

    volatile这个关键字可能很多朋友都听说过,或许也都用过.在Java 5之前,它是一个备受争议的关键字,因为在程序中使用它往往会导致出人意料的结果.在Java 5之后,volatile关键字才得以 ...

  4. volatile指令重排_面试:为了进阿里,重新翻阅了Volatile与Synchro

    面试:为了进阿里,重新翻阅了Volatile与Synchronized 在深入理解使用Volatile与Synchronized时,应该先理解明白Java内存模型 (Java Memory Model ...

  5. 再记一次止于三面的阿里面试之旅,感觉内心是拔凉拔凉的

    Hello 大家好,我是楠楠,最近心情不是很好,因为楠楠面试阿里三面挂掉了, 当收到下面这封邮件的时候猪猪内心是拔凉拔凉的.楠楠被 "Unfortunately","an ...

  6. 三面阿里失败,幸获阿里P8大牛指点,奋战三个月30*14薪入职字节

    个人背景 我的个人背景非常简单,Java开发经验3年,学历普通,一本本科计算机专业,毕业后出来就一直在一家互联网公司Crud,在公司每天重复的工作对我的技术提升并没有什么帮助: 心思开始活泛想去追寻更 ...

  7. 2022分享三面阿里:Java 面试核心手册 +Java 电子书 + 技术笔记 + 学习视频

    写在片头:声明,勿杠 首先简单说一下,这三次面试阿里并不是一次性去面的,实际上第一次面试时候还在大四,找的实习岗,不太清楚是什么部门,别问我为什么还记得面试题,有记录和复盘的习惯,再问就是杠. 个人背 ...

  8. 三 volatile关键字

    一:内存模型: 大家都知道,计算机在执行程序时,每条指令都是在CPU中执行的,而执行指令过程中,势必涉及到数据的读取和写入.由于程序运行过程中的临时数据是存放在主存(物理内存)当中的,这时就存在一个问 ...

  9. 机器学习方法三要素-阿里云大学

    机器学习方法三要素-阿里云大学 参考李航<统计学习方法>1.3章统计学习三要素 1. 机器学习方法三要素-模型.策略.算法 2. 模型 3. 策略 3.1 策略衡量指标-损失函数(单个样本 ...

最新文章

  1. Linux正变得无处不在;应用大盘点
  2. 九度OJ 区间问题
  3. asp.net2.0安全性(3)--验证与授权
  4. java如何计算html高度,如何检索HTML元素的实际宽度和高度?
  5. 利用zookeeper实现分布式服务故障自动剔除/服务自动注册的思路
  6. 博客暂停通知-------10.1~11.24
  7. leetcode42 --- trap
  8. 实验计算机控制器的实验结论,微机控制实验报告
  9. Oracle获取汉字拼音码
  10. linux无线网卡消失,Linux下无线网卡无法开启解决办法
  11. java银行账户类_使用Java编写银行账户类(面向对象思考实验)
  12. 1024,来一套程序员续命操!
  13. 【2018年11月12日】其他化学制品行业的股票估值和排名
  14. [Mysql] LAST_DAY函数 | DATE_ADD函数 | DATE_SUB函数
  15. 114个ChatGPT全网最热话题
  16. vim 常用配置整理
  17. PS初学者实例教程——图层样式的学习[斜面和浮雕]
  18. 5-旋转的小菊-旋转画布和定时器
  19. tomcat服务器一直自动关,项目在tomcat里运行一段时间总是自动崩掉的问题排查与解决...
  20. 新手入门:ST-Link和J-Link仿真器的使用

热门文章

  1. Spring beans配置方案(一) 学习笔记
  2. Spring Cloud (断路器) Hystrix(三)
  3. bzoj1426 收集邮票
  4. Android的广播接收器BroadcastReceiver
  5. git、cocoapod组件化开发常用命令
  6. 十进制四则运算计算器代码,输入为字符串
  7. Ajax之跨域访问与JSONP
  8. iOS Core Animation学习总结(2)--实现自定义图层
  9. println()函数输出int类型返回值错误的问题
  10. acm 3278(poj4001)