我个人一直认为:网络、并发相关的知识,相对其他一些编程知识点更难一些,主要是不好调试并且涉及内容太多 !

所以今天就取一篇并发相关的内容分享下,我相信大家认真看完会有收获的。

大家可以先看看这个问题,看看这个是否有问题呢? 那里有问题呢?

如果你在这个问题上面停留超过5s的话,那么表示你对这块某些知识还有点模糊,需要再巩固下,下面我们一起来分析下!

1. 结论

多线程并发的同时进行set、get操作,A线程调用set方法,B线程并不一定能对这个改变可见!!!

2. 分析

这个类非常简单,里面有一个属性,有2个方法:get、set方法,一个用来设置属性值,一个用来获取属性值,在设置属性方法上面加了synchronized。

隐式信息:多线程并发的同时进行set、get操作,A线程调用set方法,B线程可以里面感知到吗???

说到这里,问题就变成了synchronized在刚刚说的上下文下面能否保证可见性!!!

3. 关键词synchronized的用法

  • 指定加锁对象:对给定对象加锁,进入同步代码前需要获得给定对象的锁。

  • 直接作用于实例方法:相当于对当前实例加锁,进入同步代码前要获得当前实例的锁。

  • 直接作用于静态方法:相当于对当前类加锁,进入同步代码前要获得当前类的锁。

synchronized它的工作就是对需要同步的代码加锁,使得每一次只有一个线程可以进入同步块(其实是一种悲观策略)从而保证线程之间得安全性。

从这里我们可以知道,我们需要分析的属于第二类情况,也就是说多个线程如果同时进行set方法的时候,由于存在锁,所以会一个一个进行set操作,并且是线程安全的,但是get方法并没有加锁,表示假如A线程在进行set的同时B线程可以进行get操作。并且可以多个线程同时进行get操作,但是同一时间最多只能有一个set操作。

4. Java 内存模型 happens-before原则

JSR-133 内存模型使用 happens-before 的概念来阐述操作之间的内存可见性。在 JMM 中,如果一个操作执行的结果需要对另一个操作可见,那么这两个操作之间必须要存在 happens-before 关系。这里提到的两个操作既可以是在一个线程之内,也可以是在不同线程之间。

与程序员密切相关的 happens-before 规则如下:

  • 程序顺序规则:一个线程中的每个操作,happens-before 于该线程中的任意后续操作。

  • 监视器锁规则:对一个监视器的解锁,happens-before 于随后对这个监视器的加锁。

  • volatile 变量规则:对一个 volatile 域的写,happens-before 于任意后续对这个 volatile 域的读。

  • 传递性:如果 A happens-before B,且 B happens-before C,那么 A happens-before C。

注意,两个操作之间具有 happens-before 关系,并不意味着前一个操作必须要在后一个操作之前执行!happens-before 仅仅要求前一个操作(执行的结果)对后一个操作可见,且前一个操作按顺序排在第二个操作之前(the first is visible to and ordered before the second)。

其中有监视器锁规则:对一个监视器的解锁,happens-before 于随后对这个监视器的加锁。这一条,仅仅只是针对synchronized的set方法,而对于get并没有这方面的说明。

其实在这种上下文下面一个synchronized的set方法,一个普通的get方法,a线程调用set方法,b线程并不一定能对这个改变可见!

5. volatile

volatile可见性
前面happens-before原则就提到:volatile 变量规则:对一个 volatile 域的写,happens-before 于任意后续对这个 volatile 域的读。volatile从而保证了多线程下的可见性!!!

volatile 禁止内存重排序
下面是 JMM 针对编译器制定的 volatile 重排序规则表:

为了实现 volatile 的内存语义,编译器在生成字节码时,会在指令序列中插入内存屏障来禁止特定类型的处理器重排序。

下面是基于保守策略的 JMM 内存屏障插入策略:

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

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

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

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

下面是保守策略下,volatile 写操作 插入内存屏障后生成的指令序列示意图:

下面是在保守策略下,volatile 读操作 插入内存屏障后生成的指令序列示意图:

上述 volatile 写操作和 volatile 读操作的内存屏障插入策略非常保守。在实际执行时,只要不改变 volatile 写-读的内存语义,编译器可以根据具体情况省略不必要的屏障。

双重检查锁实现单例中就需要用到这个特性!!!

6. 模拟

通过上面的分析,其实这个题目涉及到的内容都提到了,并且进行了解答。

虽然你知道的原因,但是想模拟并不是一件容易的事情!,下面我们来模拟看看效果:

public class ThreadSafeCache {int result;public int getResult() {return result;}public synchronized void setResult(int result) {this.result = result;}public static void main(String[] args) {ThreadSafeCache threadSafeCache = new ThreadSafeCache();for (int i = 0; i < 8; i++) {new Thread(() -> {int x = 0;while (threadSafeCache.getResult() < 100) {x++;}System.out.println(x);}).start();}try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}threadSafeCache.setResult(200);}
}

效果:

程序会一直卡在这边不动,表示set修改的200,get方法并不可见!!!

添加volatile 关键词观察效果

其实例子中synchronized关键字可以去掉,仅仅用volatile即可。

效果:

代码很快正常结束了!

结论:多线程并发的同时进行set、get操作,A线程调用set方法,B线程并不一定能对这个改变可见!!!,上面的代码中,如果对get方法也加synchronized也是可见的,还是happens-before的监视器锁规则:对一个监视器的解锁,happens-before 于随后对这个监视器的加锁。只是volatile比synchronized更轻量级,所以本例直接用volatile。但是对于符合非原子操作i++这里还是不行的还是需要synchronized。

分享一道阿里Java并发面试题 (详细分析篇)相关推荐

  1. 阿里一道Java并发面试题 (详细分析篇)

    题目 我个人一直认为:网络.并发相关的知识,相对其他一些编程知识点更难一些,主要是不好调试并且涉及内容太多 ! 所以今天就取一篇并发相关的内容分享下,我相信大家认真看完会有收获的. 大家可以先看看这个 ...

  2. 面试java回答优缺点_阿里Java开发面经分享,面试题(附回答)

    下面要给大家分享一个阿里Java开发面经,里面包含了具体的问题,以及相关回答,主要包括了数据库.操作系统.项目相关.java基础.数据结构.计算机网络等内容. 一.数据库 1.为什么在项目中用Mong ...

  3. 推荐:全网最全的Java并发面试题及答案。

    转载自  推荐:全网最全的Java并发面试题及答案. 1.在java中守护线程和本地线程区别? java中的线程分为两种:守护线程(Daemon)和用户线程(User). 任何线程都可以设置为守护线程 ...

  4. java并发面试题(一)基础

    转载:http://ifeve.com/javaconcurrency-interview-questions-base/java并发面试题(一)基础 本文整理了常见的Java并发面试题,希望对大家面 ...

  5. 思科java开发面试题

    思科java开发面试题 1.mybatis怎么用的 2.MySQL怎么用的 3.git怎么用的 4.hashmap和hashtable 5.建立多线程 6.设计模式 7.程序的优化 8.程序的安全

  6. java消费者模式_基于Java 生产者消费者模式(详细分析)

    生产者消费者模式是多线程中最为常见的模式:生产者线程(一个或多个)生成面包放进篮子里(集合或数组),同时,消费者线程(一个或多个)从篮子里(集合或数组)取出面包消耗.虽然它们任务不同,但处理的资源是相 ...

  7. java生产线消费者,基于Java 生产者消费者模式(详细分析)

    生产者消费者模式是多线程中最为常见的模式:生产者线程(一个或多个)生成面包放进篮子里(集合或数组),同时,消费者线程(一个或多个)从篮子里(集合或数组)取出面包消耗.虽然它们任务不同,但处理的资源是相 ...

  8. Java工程师笔试题整理[校招篇]

    Java工程师笔试题整理[校招篇] 阿里巴巴 2016 阿里巴巴2016研发工程师笔试题(四) 阿里巴巴2016研发工程师笔试题(三) 阿里巴巴2016研发工程师笔试题(二) 2015 阿里巴巴201 ...

  9. 「独家」五面阿里P6:Java开发面试题及答案

    说说最近的项目 如何实现session共享,用redis如何实现 缓存击穿的概念和解决方案 说说微服务,微服务之间如何管理 二面(现场) java nio常?用的三个类 java里面的同步锁了解吗?C ...

最新文章

  1. 《结对-结对编项目作业名称-开发环境搭建过程》
  2. 毕业生必须知道:干部身份、三方协议、派遣证
  3. c++ 二分查找的函数 lower_bound upper_bound binary_search
  4. 解决ie7不支持after、before的方法
  5. centos 升级curl版本
  6. 网页视频15分钟自动暂停_在15分钟内学习网页爬取
  7. 常用英雄胜率怎么刷_单排刷英雄胜率或者炸鱼方法。
  8. windows10安装Visual Studio 2017
  9. zabbix中mysql连不上的排错_zabbix使用排错 - oschina130111的个人空间 - OSCHINA - 中文开源技术交流社区...
  10. Codeforces Round #358 (Div. 2) Alyona and Strings
  11. 抖音网上如何赚钱变现,有哪些具体的方法
  12. 全国计算机等级考点申请,全国计算机等级考试(NCRE)考点申请书某高校(367)
  13. 金山词霸电脑版 百度算法
  14. [Ynoi2016]炸脖龙I/Nephren Ruq Insania
  15. 高通CPU处理器解析
  16. 无需修改bios即可让任意主板实现NVME启动
  17. 特斯拉进化论:舍命狂奔背后的生存哲学
  18. Linux系统安装Anaconda3保姆级教程
  19. nvm安装,nvm的使用,nvm常用命令,nvm安装node报错,nvm切换不了,等系列集合
  20. cad批量选择相同块_在CAD中如何快速选择相同或类似的图形、图块

热门文章

  1. 【ubuntu+opencv3】ubuntu16.04+qt5+opencv3.2.0编译与安装
  2. macOS 中使用钥匙串保存密码
  3. 使用 Fluentd 和 ElasticSearch Stack 实现 Kubernetes 的集群 Logging
  4. XHTML 学前概述
  5. asp.net %% %#%%=%%@%%$%用法与区别
  6. 创建一个提供数据 API 的 Node.js 网站
  7. UNIX网络编程——TCP/IP简介
  8. CentOS\fedora使用yum update更新时不升级内核的方法
  9. 小心robots.txt影响蜘蛛的爬行
  10. Q学习(Q learning) 强化学习