volatile相关内容
volatile是jvm提供的轻量级的同步机制
保证可见性(一个线程的修改对其它线程是可见的)
不保证原子性
禁止指令重排序
什么是指令重排?
计算机在执行程序时,为了提高性能,编译器和处理器会对指令做重排,过程如下
源代码--》编译器优化的重排--》指令并行的重排--》内存系统的重排--》最终执行的指令
处理器在进行重排时必须考虑数据之间的依赖性
单线程环境下重拍后执行的结果与代码顺序执行的结果一致
多线程环境下线程交替进行,由于编译器优化重排的存在,两个线程使用的变量能否保证一致性无法确定/**
* 如果有两个线程
* 一个执行fun1,一个执行fun2
* 1. 当fun1中语句一和语句二执行时发生指令重排,语句二在语句一之前执行
* 那么语句二执行完后,没等语句一执行,fun2中执行,a=5
* 2. 语句一先执行时,最后a=6
* 所以多线程环境下,由于编译器的优化重排,结果不确定
* <p>
* volatile会禁止指令重排,让执行顺序按照代码编写的顺序执行
*/
class ResortedDemo {
int a;
boolean flag; public void fun1() {
a = 1;//语句一
flag = true;//语句二
} public void fun2() {
if (flag) {
a = a + 5;
System.out.println(a);
}
}
}
JMM(java memory model)java内存模型
jmm本身是一种抽象的概念,并不真实存在,它描述的是一组规范,通过这组字段定义了程序中各个变量(包括程序字段,静态字段和构成数组对象的元素)的访问方式
JMM关于同步的规定
- 线程加锁前,必须读取主内存的最新值到自己的工作内存
- 线程解锁前,必须把共享变量的值刷新回主内存
- 加锁解锁是同一把锁
由于JVM运行程序的实体是线程, 每个线程创建时JVM都会为其分配一个工作内存
线程对变量的操作必须在工作内存进行,首先将变量从主内存拷贝到自己的工作内存空间,然后对变量进行操作,然后再将变量写回到主内存 ,
不能直接操作主内存的变量,线程之间不能访问其它线程的工作内存,因此线程间的通信(传值)必须通过主内存完成
volatile代码实例
/**
* # 验证volatile的可见性(每次读取都直接从主内存读取)
* 假设number=0;
* 1. number前面没有volatile,没有可见性
* 就算第一个线程更改了number的值,并写回了主内存,但由于没有可见性
* main线程并不知道已经修改,所以main线程中number一直等于0,一直在while循环中
* 2. number前面加上volatile,保证可见性
* 第一个线程更改完值并写回主内存后,由于number的可见性,
* main线程中立刻就能读取到主内存中number的修改,跳出while循环
* # 验证volatile不保证原子性
* 1. 原子性是什么意思
* 不可分割,完整性,也就是某个线程正在做某个具体业务时中间不可以被加塞或者被分割,
* 其它线程不能对该线程获取的资源进行任何操作(包括读取)
* 要么同时成功,要么同时失败
* 2.如何验证:
* 创建20个线程,每个线程对number做1000次加一操作,若最后number=20*1000,则有原子性,反之则没有
* <p>
* 经运行程序,number<20000,所以volatile不保证原子性
* <p>
* 3. 为什么不保证原子性
* number++:分为三步骤
* 1. 从主内存取值
* 2. 对值操作加1
* 3. 写回主内存
* <p>
* 多个线程同时读取到之内存中number的值,并操作加1,
* 最后把值写回主内存过程中,可能会出现数据丢失写值的效果
* 4. 如何解决原子性
* 1. 加sync
* 2. 使用juc包下的AtomicInteger
* 使用juc包下的AtomicInteger中CAS通过在将值写入主内存时循环比较的方式保证原子性
*/
public class VolatileDemo {
public static void main(String[] args) {
A a = new A();
for (int i = 0; i < 20; i++) {
new Thread(() -> {
for (int j = 0; j < 1000; j++) {
//测试volatile修饰的变量,不保证原子性
a.number++;
//测试AtomicInteger解决原子性
a.atomicInteger.getAndIncrement();
}
}, "线程" + i).start();
}
//需要等待上面20个线程运行结束,才能运行主线程
//后台有两个默认线程,所以大于2
while (Thread.activeCount() > 2) {
//让当前线程获得可运行状态
Thread.yield();
}
System.out.println(Thread.currentThread().getName() + "number=" + a.number);
System.out.println(a.atomicInteger.get());
}
public static void verifyVisible() {
A a = new A();
//第一个线程
new Thread(() -> {
System.out.println(Thread.currentThread().getName() + "start");
try {
TimeUnit.SECONDS.sleep(3);
} catch (InterruptedException e) {
e.printStackTrace();
}
a.change();
System.out.println(Thread.currentThread().getName() + "end:" + a.number);
}).start();
//第二个线程
while (a.number == 0) {
//main线程一直在这里循环,如果a.number一直等于0,也就是没有可见性
//如果跳出了循环,就代表可见性触发了
}
System.out.println(Thread.currentThread().getName() + "end:" + a.number);
}
}
class A {
volatile int number = 0;
AtomicInteger atomicInteger = new AtomicInteger(0);
public void change() {
number = 26;
}
}
volatile相关内容的更多相关文章
- linux用户权限相关内容查看
linux用户权限相关内容查看 1 用户信息 创建用户一个名为 webuser 的账号,并填写相应的信息: root@iZ94fabhqhuZ:~# adduser webuser Adding ...
- SharePoint安全 - 在Goolge和Bing中查找SharePoint相关内容
博客地址 http://blog.csdn.net/foxdave 本篇提供两个查询串字典,分别对应Google和Bing的搜索,用来查询SharePoint网站的相关内容 Google ShareP ...
- 韩顺平细说Servlet视频系列之tom相关内容
韩顺平细说Servlet视频系列之tom相关内容 tomcat部署项目操作(注意:6.0版本以后的支持该操作,5.x版本需要另外配置?待验证!) 项目发布到tomcat的webapps文件下,然后启动 ...
- jQuery实现页内查找相关内容
当需要在页面中查找某个关键字时,一是可以通过浏览器的查找功能实现,二是可以通过前端脚本准确查找定位,本文介绍通过jQuery实现的页面内容查找定位的功能,并可扩展显示查找后的相关信息. 本文以查找车站 ...
- Struts2(四)——页面相关内容
上篇博客总结了数据流转各个方面的内容,这篇重点说一下框架对于界面上知识. 一,说到页面,记得在总体介绍中,说到Struts2比Struts1的一方面优势就是它支持更多的视图技术(Freemarker, ...
- 学习笔记之html5相关内容
写一下昨天学习的html5的相关内容,首先谈下初次接触html5的感受.以前总是听说html5是如何的强大,如何的将要改变世界.总是充满了神秘感.首先来谈一下我接触的第一个属性是 input的里面的 ...
- 基于KNN的相关内容推荐
如果做网站的内容运营,相关内容推荐可以帮助用户更快地寻找和发现感兴趣的信息,从而提升网站内容浏览的流畅性,进而提升网站的价值转化.相关内容 推荐最常见的两块就是“关联推荐”和“相关内容推荐”,关联推荐 ...
- 第一天上午——HTML网页基础知识以及相关内容
今天上午学习了HTML基础知识以及相关内容,还有DW的基本使用方法. HTML(HyperText Markup Language):超文本标记语言,超文本:网页中除了包含文本文字之外,还包含了图片, ...
- python爬虫主要就是五个模块:爬虫启动入口模块,URL管理器存放已经爬虫的URL和待爬虫URL列表,html下载器,html解析器,html输出器 同时可以掌握到urllib2的使用、bs4(BeautifulSoup)页面解析器、re正则表达式、urlparse、python基础知识回顾(set集合操作)等相关内容。
本次python爬虫百步百科,里面详细分析了爬虫的步骤,对每一步代码都有详细的注释说明,可通过本案例掌握python爬虫的特点: 1.爬虫调度入口(crawler_main.py) # coding: ...
随机推荐
- django中app分组
08.13自我总结 django中app分组 一.django路由系统app进行分组 1.创建app 使用pycharm创建django的时候, 加上app的名字,后续多个app只需复制粘贴之前app ...
- drf框架serializers中ModelSerializer类简化序列化和反序列化操作
0905自我总结 drf框架serializers中ModelSerializer类 基于seriallizer类进行简化 https://www.cnblogs.com/pythonywy/p/11 ...
- phpstudy后门rce批量利用脚本
写两个 一个批量检测的 一个交互式shell的 暂时py 图形化的qt写出来..有点问题 后门包 : GET / HTTP/1.1Host: 127.0.0.1User-Agent: Mozill ...
- C++ set 用法略解
先看一段代码. #include<iostream> #include<set> #include<cstdio> #include<cstdlib> ...
- [JOYOI1510] 专家复仇 - Floyd
题目限制 时间限制 内存限制 评测方式 题目来源 1000ms 131072KiB 标准比较器 Local 题目背景 外星人完成对S国的考察后,准备返回,可他们的飞碟已经没燃料了……S国的专家暗自窃喜 ...
- Halcon一日一练:图像分割之基本概念
1.什么是图像分割: 图像分割就是把图像中特定的目标提出来,进行处理. 2.为什么要做图像分割: 图像分割是由图像处理到图像分析的关键步骤,准确的来说,没有图像分割,图像处理将无法实现其后续的操作.进 ...
- 关于git提交
16年申请注册的git账号,除了需要找一些有用的demo时会逛一下,其实一直没拿起来真实用过. 好了,说一下今天下午呕心沥血弄出来的东西,其实就是简单的用git小黑窗口推文件夹上去,本人较笨,会按照我 ...
- xshell rz commend not found
sudo apt-get install lrzsz 上传rz 下载sz
- 写了那么久的Python,你应该学会使用yield关键字了
写过一段时间代码的同学,应该对这一句话深有体会:程序的时间利用率和空间利用率往往是矛盾的,可以用时间换空间,可以用空间换时间,但很难同时提高一个程序的时间利用率和空间利用率. 但如果你尝试使用生成器来 ...
- vue 开发插件流程
UI demo UI 插件汇总 我的github iSAM2016 在练习写UI组件的,用到全局的插件,网上看了些资料.看到些的挺好的,我也顺便总结一下写插件的流程: 声明插件-> 写插件-&g ...