点击上方“方志朋”,选择“设为星标”

回复”666“获取新整理的面试文章

来源:hapjin

cnblogs.com/hapjin/p/5492880.html

一,volatile关键字的可见性

要想理解volatile关键字,得先了解下JAVA的内存模型,Java内存模型的抽象示意图如下:

从图中可以看出:

①每个线程都有一个自己的本地内存空间--线程栈空间???线程执行时,先把变量从主内存读取到线程自己的本地内存空间,然后再对该变量进行操作

②对该变量操作完后,在某个时间再把变量刷新回主内存

因此,就存在内存可见性问题,看一个示例程序:(摘自书上)

public class RunThread extends Thread {private boolean isRunning = true;public boolean isRunning() {return isRunning;}public void setRunning(boolean isRunning) {this.isRunning = isRunning;}@Overridepublic void run() {System.out.println("进入到run方法中了");while (isRunning == true) {}System.out.println("线程执行完成了");}
}public class Run {public static void main(String[] args) {try {RunThread thread = new RunThread();thread.start();Thread.sleep(1000);thread.setRunning(false);} catch (InterruptedException e) {e.printStackTrace();}}
}

Run.java 第28行,main线程 将启动的线程RunThread中的共享变量设置为false,从而想让RunThread.java 第14行中的while循环结束。

如果,我们使用JVM -server参数执行该程序时,RunThread线程并不会终止!从而出现了死循环!!

原因分析:

现在有两个线程,一个是main线程,另一个是RunThread。它们都试图修改 第三行的 isRunning变量。按照JVM内存模型,main线程将isRunning读取到本地线程内存空间,修改后,再刷新回主内存。

而在JVM 设置成 -server模式运行程序时,线程会一直在私有堆栈中读取isRunning变量。因此,RunThread线程无法读到main线程改变的isRunning变量

从而出现了死循环,导致RunThread无法终止。这种情形,在《Effective JAVA》中,将之称为“活性失败”

解决方法,在第三行代码处用 volatile 关键字修饰即可。这里,它强制线程从主内存中取 volatile修饰的变量。

volatile private boolean isRunning = true;

扩展一下,当多个线程之间需要根据某个条件确定 哪个线程可以执行时,要确保这个条件在 线程 之间是可见的。因此,可以用volatile修饰。

综上,volatile关键字的作用是:使变量在多个线程间可见(可见性)

二,volatile关键字的非原子性

所谓原子性,就是某系列的操作步骤要么全部执行,要么都不执行。

比如,变量的自增操作 i++,分三个步骤:

  • 从内存中读取出变量 i 的值

  • 将 i 的值加1

  • 将 加1 后的值写回内存

这说明 i++ 并不是一个原子操作。因为,它分成了三步,有可能当某个线程执行到了第②时被中断了,那么就意味着只执行了其中的两个步骤,没有全部执行。

关于volatile的非原子性,看个示例:

public class MyThread extends Thread {public volatile static int count;private static void addCount() {for (int i = 0; i < 100; i++) {count++;}System.out.println("count=" + count);}@Overridepublic void run() {addCount();}
}public class Run {public static void main(String[] args) {MyThread[] mythreadArray = new MyThread[100];for (int i = 0; i < 100; i++) {mythreadArray[i] = new MyThread();}for (int i = 0; i < 100; i++) {mythreadArray[i].start();}}
}

MyThread类第2行,count变量使用volatile修饰

Run.java 第20行 for循环中创建了100个线程,第25行将这100个线程启动去执行 addCount(),每个线程执行100次加1

期望的正确的结果应该是 100*100=10000,但是,实际上count并没有达到10000

原因是:volatile修饰的变量并不保证对它的操作(自增)具有原子性。(对于自增操作,可以使用JAVA的原子类AutoicInteger类保证原子自增)

比如,假设 i 自增到 5,线程A从主内存中读取i,值为5,将它存储到自己的线程空间中,执行加1操作,值为6。此时,CPU切换到线程B执行,从主从内存中读取变量i的值。由于线程A还没有来得及将加1后的结果写回到主内存,线程B就已经从主内存中读取了i,因此,线程B读到的变量 i 值还是5

相当于线程B读取的是已经过时的数据了,从而导致线程不安全性。这种情形在《Effective JAVA》中称之为“安全性失败”

综上,仅靠volatile不能保证线程的安全性。(原子性)

此外,volatile关键字修饰的变量不会被指令重排序优化。这里以《深入理解JAVA虚拟机》中一个例子来说明下自己的理解:

线程A执行的操作如下:

Map configOptions ;
char[] configText;volatile boolean initialized = false;//线程A首先从文件中读取配置信息,调用process...处理配置信息,处理完成了将initialized 设置为true
configOptions = new HashMap();
configText = readConfigFile(fileName);
processConfig(configText, configOptions);//负责将配置信息configOptions 成功初始化
initialized = true;

线程B等待线程A把配置信息初始化成功后,使用配置信息去干活…..线程B执行的操作如下:

while(!initialized)
{sleep();
}//使用配置信息干活
doSomethingWithConfig();

如果initialized变量不用 volatile 修饰,在线程A执行的代码中就有可能指令重排序。在 后端技术精选 公众号回复“后端面试”,获取最新面试资料。

即:线程A执行的代码中的最后一行:initialized = true 重排序到了 processConfig方法调用的前面执行了,这就意味着:配置信息还未成功初始化,但是initialized变量已经被设置成true了。那么就导致 线程B的while循环“提前”跳出,拿着一个还未成功初始化的配置信息去干活(doSomethingWithConfig方法)。。。。

因此,initialized 变量就必须得用 volatile修饰。这样,就不会发生指令重排序,也即:只有当配置信息被线程A成功初始化之后,initialized 变量才会初始化为true。综上,volatile 修饰的变量会禁止指令重排序(有序性)

三,volatile 与 synchronized 的比较

volatile主要用在多个线程感知实例变量被更改了场合,从而使得各个线程获得最新的值。它强制线程每次从主内存中讲到变量,而不是从线程的私有内存中读取变量,从而保证了数据的可见性。

比较:

①volatile轻量级,只能修饰变量。synchronized重量级,还可修饰方法

②volatile只能保证数据的可见性,不能用来同步,因为多个线程并发访问volatile修饰的变量不会阻塞。

synchronized不仅保证可见性,而且还保证原子性,因为,只有获得了锁的线程才能进入临界区,从而保证临界区中的所有语句都全部执行。多个线程争抢synchronized锁对象时,会出现阻塞。

四,线程安全性

线程安全性包括两个方面,①可见性。②原子性。

从上面自增的例子中可以看出:仅仅使用volatile并不能保证线程安全性。而synchronized则可实现线程的安全性。

热门内容:干掉GuavaCache:Caffeine才是本地缓存的王
稳~阿里程序员常用的 15 款开发者工具四连问:API 接口应该如何设计?如何保证安全?如何签名?如何防重?
再见,FastJson...技巧:MyBatis 中的trim标签,好用!
求求你们了,别再写满屏的 try catch 了!!最近面试BAT,整理一份面试资料《Java面试BAT通关手册》,覆盖了Java核心技术、JVM、Java并发、SSM、微服务、数据库、数据结构等等。获取方式:点“在看”,关注公众号并回复 666 领取,更多内容陆续奉上。
明天见(。・ω・。)ノ♡

JAVA多线程之volatile 与 synchronized 的比较相关推荐

  1. Java多线程之volatile详解

    Java多线程之volatile详解 目录: 什么是volatile? JMM内存模型之可见性 volatile三大特性之一:保证可见性 volatile三大特性之二:不保证原子性 volatile三 ...

  2. java多线程之volatile理解

       最近一直在看多线程的一些知识,看了一些书和一些博客,收获还是挺多的,最近看了<java并发编程的艺术>这本书感觉收获很大也推荐给各位,同时也结合以前看的博客就好好的总结一下自己所学的 ...

  3. Java多线程之Synchronized和Lock的区别

    Java多线程之Synchronized和Lock的区别 目录: 原始构成 使用方法 等待是否可以中断 加锁是否公平 锁绑定多个条件Condition 小结:Lock相比较Synchronized的优 ...

  4. Java多线程之CAS深入解析

    Java多线程之CAS深入解析 目录: CAS是什么 CAS底层原理Unsafe深入解析 CAS缺点 引子:蚂蚁花呗一面:讲一讲AtomicInteger,为什么要用CAS而不是synchronize ...

  5. JAVA多线程之wait/notify

    本文主要学习JAVA多线程中的 wait()方法 与 notify()/notifyAll()方法的用法. ①wait() 与 notify/notifyAll 方法必须在同步代码块中使用 ②wait ...

  6. Java多线程之8Lock问题解析

    Java多线程之8Lock问题解析 本文目录 1. 8Lock实例: 标准访问的时候,请问先打印邮件还是短信? sendEmail方法暂停4秒钟,请问先打印邮件还是短信? 新增Hello普通方法,请问 ...

  7. java多线程之wait和notify协作,生产者和消费者

    这篇直接贴代码了 package cn.javaBase.study_thread1;class Source {public static int num = 0; //假设这是馒头的数量 }cla ...

  8. Java多线程之Callable、Future和FutureTask

    Java多线程之Callable接口 自己想总结一下的,看到一篇总结的更好的博客,就转载了,突然感觉真轻松,哈哈哈哈 文章转载于:Matrix海子:Java并发编程:Callable.Future和F ...

  9. Java多线程之CAS缺点

    Java多线程之CAS缺点 目录: 循环时间开销很大 只能保证一个共享变量的原子操作 引来ABA问题及解决方案(重点) 1. 循环时间开销很大 通过看源码,我们发现有个do while,如果CAS失败 ...

最新文章

  1. 软件开发环境-环境信息库
  2. framework之Activity启动流程(基于Android11源码)
  3. 漫谈流式计算的一致性
  4. nginx收到空包问题
  5. 天天都会写接口,但它的用途和好处有多少人能说得清楚?
  6. python Hbase Thrift pycharm 及引入包
  7. 三维重建学习(5):简单地从数学原理层面理解双目立体视觉
  8. LAMP架构介绍,MySQL、MariaDB介绍,MySQL安装
  9. linux 查找文件夹_用python打造一个基于socket的文件(夹)传输系统
  10. springboot 的 RedisTemplate 的 execute 和 executePipelined 功能的区别redis
  11. eclipse中误删除的java文件 代码如何恢复
  12. iOS:fishhook原理分析
  13. Kaggle_Predict Future Sales_Prac 1(时间序列预测商品销量)
  14. VapourSynth压制教程 【Waifu2x N卡 放大教程】
  15. word目录的制表符前导符无法改变
  16. TGRS2022/遥感:An Empirical Study of Remote Sensing Pretraining遥感预训练的实证研究
  17. snv服务器备份方案
  18. PLSQL查看所连接的ORACLE的版本
  19. STL:string容器特性、定义、初始化、等号、取值、拼接、查找、替换、比较、字串、插入、删除
  20. Android12 支持高斯模糊及高斯模糊原理

热门文章

  1. [Ahoi2008]Meet 紧急集合
  2. 通过web sql实现增删查改
  3. 微软企业库4.1学习笔记(八)创建对象 续集2
  4. 【青少年编程】【四级】绘图程序优化
  5. Numpy入门教程:12. 线性代数
  6. 【ACM】家喻户晓的中药店(待更)
  7. 马斯克公开支持“上班摸鱼”:让工作更愉快!
  8. 首届“陇剑杯”网络安全大赛线上赛圆满结束
  9. 基于PyTorch,如何构建一个简单的神经网络
  10. 在 5G 速度上,iPhone 12 只是个弟弟