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. DIV或者DIV里面的图片水平与垂直居中的方法

    <div class=“box”> <img /> </div> 水平居中的常用方式: text-align:center ——这可以实现子元素字体,图片的水平居中 ...

  2. IntelliJ IDEA和Eclipse最常用的快捷键对应表:

    描述 Eclipse Intellij 代码补全 Ctrl+space Ctrl+space 打开类或者接口 (两个IDE都支持使用“驼峰字符”前缀的方式来过滤查找列表,进而轻松完成搜索:比如:可 以 ...

  3. 如何使用无线连接来使Android调试手机

    进入Android Studio.(我的是2.2版本) File->Setting->Plugins Browse repositories... 搜索 ADB WIFI 并安装 重启An ...

  4. LL(1)文法分析表的构造和分析过程示例

    在考完编译原理之后才弄懂,悲哀啊.不过懂了就好,知识吗,不能局限于考试. 文法: E→TE' E'→+TE'|ε T→FT ' T'→*FT'|ε F→id| (E) 一.首先判断是不是 LL(1)文 ...

  5. 淘淘商城之SSM框架整合概要

    一.后台系统所用的技术 1)框架:Spring + SpringMVC + Mybatis: 2)前端:EasyUI: 3)数据库:mysql 二.创建数据库 1)安装mysql数据库: 2)在mys ...

  6. ssh 登录出现Are you sure you want to continue connecting (yes/no)?解决方法

    ssh 登录出现Are you sure you want to continue connecting (yes/no)?解决方法 1,可以使用ssh -o 的参数进行设置例如: ssh -o St ...

  7. android 简单的读写联系人

    一.读取联系人 (1).从raw_contact表读 contact_id (2).从data表中读取data1 (3).从mimetypes读取mimetype 1.AndroidManifest. ...

  8. 近几年杭电OJ大型比赛题目合集【更新到2017年11月初】

    2017年: 区域赛网络赛   6194~6205    6206~6216 区域赛网络赛   6217~6229 2016年: 区域赛网络赛  5868~5877    5878~5891    5 ...

  9. MySQL自动设置create_time和update_time

    参考表结构 CREATE TABLE `t_baby` ( `baby_id` int(11) NOT NULL AUTO_INCREMENT, `user_id` int(11) NOT NULL, ...

  10. pygame(class类)调用视图的方法

    以下将介绍pygame精灵动画的基础知识,希望对大家有帮助:1.在此,精灵类必须继承pygame.sprite.Sprite并初始化pygame.sprite.Sprite.__init__(self ...