volatile关键字

volatile关键字是什么

在上一章我们讲到了并发的的三个概念,那么今天在讲解下在java中可以保证可见性和有序性的一个关键字。

volatile关键字 :当变量的值被该关键字修饰后该值任何读写操作对于其他线程是立即可见的。并且被关键字修饰后的变量被禁止重排序。

volatile原理解析

在定义中我们可以看到volatile关键字有2个特性,可见性和有序性,那么volatile是如何保证这个可见性和有序性的呢?那他为什么不能保证原子性呢?

首先volatile通过加入内存屏障和禁止指令重排序优化来实现的

可见性

写操作

对volatile 变量进行写操作时,会在写操作后添加一个Store Memory Barrier屏障指令将工作内存中的变量写入主内存中。该指令会优先执行在缓冲区所有该变量的相关操作。相当于手动刷新到主内存。

读操作

对volatile变量进行读操作时,会在读操作之前添加一个Load Memory Barrier屏障指令,从主内存读取共享变量。该指令会将失效队列中所有的指令执行,让其他cpu对该变量的改动全部生效,并且刷新回主内存,然后在读取该主内存中的值。相当于手动刷新该变量的最新值。

有序性

  1. StoreStore屏障 该指令之前的写操作不能和该指令之后的写操作重排序.
  2. StoreLoad屏障 该指令之前的写操作和该指令之后的读操作重排序.
  3. LoadLoad屏障 该指令之前的读操作不能和该指令之后的读操作重排序.
  4. LoadStore屏障 该指令之前的读操作不能和该指令之后的写操作重排序。

写操作

在volatile写和普通写不能交换位置。也就能保证volatile写的值是最新的值。

volatile写以及后来的读不能交换位置,也就是后来的读必须在volatile之后执行。

读操作

而在volatile读不能与在volatile读之后的读操作和写操作交换位置。也就是说volatile读之后的读写操作都必须在volatile读之后完成。

注意:Memory Barrier 可以参看CPU缓存一致性协议MESI-硬件内存模型

volatile适用场景

例子一

@Slf4j
public class UnsafeExample {
private static final int CLIENT_TOTAL = 30000;
private static final int THREAD_TOTAL = 300;
private static volatile int count = 0; public static void main(String[] args) throws InterruptedException {
//模拟并发
ExecutorService executorService = Executors.newCachedThreadPool();
CountDownLatch countDownLatch = new CountDownLatch(CLIENT_TOTAL);
Semaphore semaphore = new Semaphore(THREAD_TOTAL);
for(int i = 0;i<CLIENT_TOTAL;i++){
executorService.execute(()->{
try{
semaphore.acquire();
add();
semaphore.release();
}catch (Exception e){
e.printStackTrace();
log.error(e.getMessage(),e);
}
countDownLatch.countDown();
});
}
countDownLatch.await();
System.out.println("统计次数:"+count);
} private static void add(){
count++;
}
}

通过测试得出volatile并不能保证上述demo的线程安全。也就是说依赖当前值来决定下一个值的场景并不适合volatile

通过上述描述可以总结出两点使用场景。

  1. 对该变量的操作不依赖当前值。(读自己 写自己)
  2. 该变量没有包含在具有其他变量的不变式中。 (读别人 写自己)

举个简单的例子

//这种volatile的使用就是错误的 违反了第一条
volatile int x = x+1; int a = 1;
//这种volatile使用场景是错误的 违反了第二条
volatile int b = a + 1;

实际上,这些条件表明,可以被写入 volatile 变量的这些有效值独立于任何程序的状态,包括变量的当前状态。

因此特别适合作为状态标记量。

使用场景

1.状态标记

重新举一下在上一章中的例子



@Slf4j
public class SimpleHappenBefore { private static int a = 0;
private static boolean flag = false; public static void main(String[] args) throws InterruptedException {
for (int i = 0; i < 1000; i++) {
ThreadA threadA = new ThreadA();
ThreadB threadB = new ThreadB();
threadA.start();
threadB.start();
threadA.join();
threadB.join();
a = 0;
flag = false;
}
} static class ThreadA extends Thread {
@Override
public void run() {
a = 1;
flag = true;
}
} static class ThreadB extends Thread {
@Override
public void run() {
if (flag) {
a = a * 1;
}
if (a == 0) {
System.out.println("ha,a==0");
}
}
}
}

上面的例子中可以看出flag的改变与其他状态 和他自己本身的状态完全没有关系。所以这里使用volatile关键字是合格的。

2.线程安全的单例对象发布

双重检查单例模式方式发布对象

单例模式-懒汉单例

public class LazySingleton {

    private static volatile LazySingleton lazySingleton;

    private LazySingleton() {
}
public static LazySingleton getInstance() {
if (lazySingleton == null) {
synchronized(LazySingleton.class) {
if (lazySingleton == null) {
lazySingleton = new LazySingleton();
}
}
}
return lazySingleton;
}
}

3.独立观察(independent observation)

比如论坛常用的统计最后一个注册的用户。

public class UserRegister {
public volatile String lastRegisterUser; public void registerUser(String user, String password) {
User u = new User();
u.setUser(user);
u.setPassword(password);
userService.registerUser(u);
lastRegisterUser = user;
}
}

4.开销较低的读-写锁策略

读操作使用volatile 写操作使用synchronzied

public class Counter {

    private volatile int value;

    public int getValue() { return value; }

    public synchronized int increment() {
return value++;
}
}

总结

volatile关键字是一个保证可见性和有序性的关键字。该关键字修饰的变量操作与该变量本身状态以及其他变量状态无关的情况下使用才可以保证其并发安全性。顺便吐槽一下(该变量没有包含在具有其他变量的不变式中)这句话,楼主理解这句话理解了很久,后面突然就顿悟了。对于语言文字的理解能力真的很重要。

并发之volatile关键字的更多相关文章

  1. Java 高效并发之volatile关键字解析

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

  2. Java并发之volatile关键字

    引言 说到多线程,我觉得我们最重要的是要理解一个临界区概念. 举个例子,一个班上1个女孩子(临界区),49个男孩子(线程),男孩子的目标就是这一个女孩子,就是会有竞争关系(线程安全问题).推广到实际场 ...

  3. Java并发之(1):volatile关键字(TIJ21-21.3.3 21.3.4)

    Java并发Java服务器端编程的一项必备技能. ** 1 简介    volatile是java中的一个保留关键字,它在英语中的含义是易变的,不稳定的.volatile像final.static等其 ...

  4. 全面理解Java内存模型(JMM)及volatile关键字(转载)

    关联文章: 深入理解Java类型信息(Class对象)与反射机制 深入理解Java枚举类型(enum) 深入理解Java注解类型(@Annotation) 深入理解Java类加载器(ClassLoad ...

  5. 全面理解Java内存模型(JMM)及volatile关键字

    [版权申明]未经博主同意,谢绝转载!(请尊重原创,博主保留追究权) http://blog.csdn.net/javazejian/article/details/72772461 出自[zejian ...

  6. 全面理解Java内存模型(JMM)及volatile关键字(转)

    原文地址:全面理解Java内存模型(JMM)及volatile关键字 关联文章: 深入理解Java类型信息(Class对象)与反射机制 深入理解Java枚举类型(enum) 深入理解Java注解类型( ...

  7. volatile关键字的详解-并发编程的体现

    xl_echo编辑整理,欢迎转载,转载请声明文章来源.欢迎添加echo微信(微信号:t2421499075)交流学习. 百战不败,依不自称常胜,百败不颓,依能奋力前行.--这才是真正的堪称强大!! 参 ...

  8. Java并发编程:volatile关键字解析

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

  9. volatile关键字 学习记录2

    public class VolatileTest2 implements Runnable{ volatile int resource = 0; public static void main(S ...

随机推荐

  1. ELK技术实战-安装Elk 5.x平台

    ELK技术实战–了解Elk各组件   转载  http://www.ywnds.com/?p=9776 ELK技术实战-部署Elk 2.x平台 ELK Stack是软件集合Elasticsearch. ...

  2. 安装使用hibernate tools

    help-Eclipse marketplace-hibernate(搜索)-jboss tools(安装)

  3. JUnit报错 java.lang.Exception:No tests found matching

    将 @RunWith(SpringRunner.class)@SpringBootTestpublic class BusinessTest { @Test public void getList() ...

  4. C# 中对于json的解析小结

    1.解析之前的json格式 [{ "Name": "冯111", "PID": "130627199202283306" ...

  5. IoC之Spring.Net在Mvc项目中的使用

    MVC中使用Spring.net 前面学习了使用Autofac来实现控制反转,这里简单记录Spring.Net实现IoC和DI的步骤 第一步:安装如下Nuget包 (Spring.Web.Mvc) i ...

  6. oracle 递归和connect by【转】

    oracle递归查询(单表包含多级上下级关系) http://www.cnblogs.com/walk-the-Line/p/4882866.html -- 查找所有第一层(包括)下的所有子节点 -- ...

  7. GYM 101173 K.Key Knocking(构造)

    原题链接 参考自 问题描述:一个长度为3*n的01串,每次可以翻转连续的两个字符,要求至多翻转n次使得这个3*n的串至少有2*n个连续的段且相邻两端不一样(就是连续的0算一段,然后连续的1,…) 解法 ...

  8. UVALive 4254 Processor(二分)

    题目链接 题意 有n个任务,每个任务有三个参数ri,di和wi,表示必须在时刻[ri,di]之内执行,工作量为wi.处理器执行速度可以变化,当执行速度为s时,工作量为wi.处理器的速度可以变化,当执行 ...

  9. cdqz2017-test8-Tree(点分树)

    n个点的带点权带边权的树,设点权为a[i],边权为b[i] 一棵树有n*(n-1)/2个点对, 定义这棵树的价值为任意两点对的(a[x]^a[y])*dis(x,y) 有m次修改一个点的点权的操作 输 ...

  10. FlowLayout OnSizeChanged

    在FlowLayout里加了20个控件,当窗口变化时,改变这20个控件的宽高,结果发现在直接点最大化时, 计算不正确导致自身的滚动条出不来.把改变大小的代码直接添加Form窗口的onSizeChagn ...