转自:http://www.cnblogs.com/aigongsi/archive/2012/04/01/2429166.html

在java线程并发处理中,有一个关键字volatile的使用目前存在很大的混淆,以为使用这个关键字,在进行多线程并发处理的时候就可以万事大吉。

Java语言是支持多线程的,为了解决线程并发的问题,在语言内部引入了 同步块 和 volatile 关键字机制。

synchronized

同步块大家都比较熟悉,通过 synchronized 关键字来实现,所有加上synchronized 和 块语句,在多线程访问的时候,同一时刻只能有一个线程能够用

synchronized 修饰的方法 或者 代码块。

volatile

用volatile修饰的变量,线程在每次使用变量的时候,都会读取变量修改后的最的值。volatile很容易被误用,用来进行原子性操作。

下面看一个例子,我们实现一个计数器,每次线程启动的时候,会调用计数器inc方法,对计数器进行加一

执行环境——jdk版本:jdk1.6.0_31 ,内存 :3G   cpu:x86 2.4G

public class Counter {

public static int count = 0;

public static void inc() {

//这里延迟1毫秒,使得结果明显

try {

Thread.sleep(1);

} catch (InterruptedException e) {

}

count++;

}

public static void main(String[] args) {

//同时启动1000个线程,去进行i++计算,看看实际结果

for (int i = 0; i < 1000; i++) {

new Thread(new Runnable() {

@Override

public void run() {

Counter.inc();

}

}).start();

}

//这里每次运行的值都有可能不同,可能为1000

System.out.println("运行结果:Counter.count=" + Counter.count);

}

}

运行结果:Counter.count=995

实际运算结果每次可能都不一样,本机的结果为:运行结果:Counter.count=995,可以看出,在多线程的环境下,Counter.count并没有期望结果是1000

很多人以为,这个是多线程并发问题,只需要在变量count之前加上volatile就可以避免这个问题,那我们在修改代码看看,看看结果是不是符合我们的期望

public classCounter {public volatile static int count = 0;public static voidinc() {//这里延迟1毫秒,使得结果明显

try{

Thread.sleep(1);

}catch(InterruptedException e) {

}

count++;

}public static voidmain(String[] args) {//同时启动1000个线程,去进行i++计算,看看实际结果

for (int i = 0; i < 1000; i++) {new Thread(newRunnable() {

@Overridepublic voidrun() {

Counter.inc();

}

}).start();

}//这里每次运行的值都有可能不同,可能为1000

System.out.println("运行结果:Counter.count=" +Counter.count);

}

}

运行结果:Counter.count=992

运行结果还是没有我们期望的1000,下面我们分析一下原因

在 java 垃圾回收整理一文中,描述了jvm运行时刻内存的分配。其中有一个内存区域是jvm虚拟机栈,每一个线程运行时都有一个线程栈,

线程栈保存了线程运行时候变量值信息。当线程访问某一个对象时候值的时候,首先通过对象的引用找到对应在堆内存的变量的值,然后把堆内存

变量的具体值load到线程本地内存中,建立一个变量副本,之后线程就不再和对象在堆内存变量值有任何关系,而是直接修改副本变量的值,

在修改完之后的某一个时刻(线程退出之前),自动把线程变量副本的值回写到对象在堆中变量。这样在堆中的对象的值就产生变化了。下面一幅图

描述这写交互

read and load 从主存复制变量到当前工作内存

use and assign  执行代码,改变共享变量值

store and write 用工作内存数据刷新主存相关内容

其中use and assign 可以多次出现

但是这一些操作并不是原子性,也就是 在read load之后,如果主内存count变量发生修改之后,线程工作内存中的值由于已经加载,不会产生对应的变化,所以计算出来的结果会和预期不一样

对于volatile修饰的变量,jvm虚拟机只是保证从主内存加载到线程工作内存的值是最新的

例如假如线程1,线程2 在进行read,load 操作中,发现主内存中count的值都是5,那么都会加载这个最新的值

在线程1堆count进行修改之后,会write到主内存中,主内存中的count变量就会变为6

线程2由于已经进行read,load操作,在进行运算之后,也会更新主内存count的变量值为6

导致两个线程及时用volatile关键字修改之后,还是会存在并发的情况。

java中volatile关键字的含义_java中volatile关键字的含义相关推荐

  1. java中volatile的含义_java中volatile关键字的含义

    (支持原创)本篇文章非原创,原文章地址: http://www.cnblogs.com/aigongsi/archive/2012/04/01/2429166.html 在java线程并发处理中,有一 ...

  2. java中volatile关键字的含义_Java里volatile关键字是什么意思

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

  3. java中实现具有传递性吗_Java中volatile关键字详解,jvm内存模型,原子性、可见性、有序性...

    一.Java内存模型 想要理解volatile为什么能确保可见性,就要先理解Java中的内存模型是什么样的. Java内存模型规定了所有的变量都存储在主内存中.每条线程中还有自己的工作内存,线程的工作 ...

  4. JAVA中的this怎么理解_Java中的this关键字怎么理解?

    一.this关键字主要有三个应用: (1)this调用本类中的属性,也就是类中的成员变量: (2)this调用本类中的其他方法: (3)this调用本类中的其他构造方法,调用时要放在构造方法的首行. ...

  5. java的对象是什么意思_Java中对象和对象引用的区别,引用、指向是什么意思

    Java的变量分为两大类:基本数据类型和引用数据类型. 其中基本类型变量有四类8种:byte short int long float double char boolean,除了8种基本数据类型变量 ...

  6. java中的case1怎么说_Java 中的 CAS 简述及原理解析

    一.CAS 是什么? CAS(Compare And Swap),比较并交换,它是一条CPU并发原语.它的功能是判断内存某个位置的值是否为预期值,如果是则更新为新的值,这个过程是原子的.1 publi ...

  7. java中容易混淆的方法_java中容易混淆的区别

    本文会随时更新一些java中容易混淆的关键字或者知识点,如有偏见之处,望留言! final和static的差别: 1,final的英语意思"最后的",在java中修饰类,方法和变量 ...

  8. java中string类面试题_java中String类的面试题大全含答案

    1.下面程序的运行结果是()(选择一项) String str1="hello"; String str2=new String("hello"); Syste ...

  9. java中的跳转语句_java中的跳转语句

    java中的跳转语句有两种:break和continue: 一.break语句 1. break语句适用于任何循环控制结构,从而是循环立即终止.当执行流程遇到break语句时,不管循环终止变量,或者布 ...

最新文章

  1. python从网上获取数据失败怎么解决_求助:scrapy爬取数据失败,反复调试都不成功...
  2. 如何开启匿名访问SharePoint 2010里的Client Object Model
  3. JSON支持什么对象/类型?
  4. WP7之题样式与数据绑定
  5. VMWare ubuntu虚拟机每次开机要等待1分30秒解决方案(A start job is running for dev-disk-by\x2duui...)(修改真实swap UUID)
  6. python安装失败如何卸载干净_彻底卸载Python环境及安装的Python模块
  7. CASE_01 基于FPGA的交通灯控制器
  8. C语言经典算法100例-002-数轴的使用
  9. 查看linux服务器内存使用情况,不够时创建Swap、手动 cached
  10. 如何使用利用LaTeX制作个人简历
  11. strace ltrace使用
  12. 用户画像pdf_【金猿产品展】易观方舟:智能用户数据中台
  13. 第1章 信息系统综合知识
  14. 20155313 杨瀚 《网络对抗技术》实验九 Web安全基础
  15. 智库大会 | 高端对话:智能科技推动管理学变革...
  16. SAP R3 系统技术基础
  17. html 截取指定字符串长度,前端js截取指定长度个数字符 v2.0.0
  18. 【模拟】【NOIP2008】笨小猴 word
  19. Dubbo3 落地实践及 Mesh 解决方案
  20. 用于定义图像热区的html标记是,html 图像热区链接

热门文章

  1. Failed to connect to 127.0.0.1 port 1080: Connection refused package:git
  2. 这样才是代码管理和 Commit 的正确姿势 | 研发效能提升36计
  3. 维大杀器来了,未来云上服务器或将实现无人值守
  4. 阿里云机器学习怎么玩?这本新手入门指南揭秘了!
  5. 笑联 x mPaaS | 12 个模块,全面小程序化,如何打造真正的一次开发复用多端?
  6. 杭州湾跨海大桥视频上云,夯实智慧高速“云基建
  7. 对话阿里云MVP裔隽跨界半生,不改赤子心
  8. 灵魂发明家自述:我就是靠这个创业成功的
  9. 同学,要不要来挑战双11零点流量洪峰?
  10. Node.js 应用故障排查手册 —— Node.js 性能平台使用指南