java volatile线程可见_Java多线程之可见性之volatile

可见性

一个线程对主内存的修改可以及时被其它线程观察到

导致共享变量在线程间不可见的原因

线程交叉执行

指令重排序加上线程交叉执行

共享变量更新后的值没有在工作内存与主存间及时更新

保证可见性和原子性

对于可见性Java提供了synchonized和volatile

volatile

通过加入内存屏障和禁止重排序优化来实现,保证可见性不保证原子性

对volatile变量进行写操作时,会在写操作后加入一条store屏障指令,将工作内存变量值刷新到主内存。

a2509f614e10d2eef4a7c530eeca8b9d.png

对volatile变量进行读操作时,会在读操作前加入一条load屏障指令,从主内存读取共享变量。

871eb2bdf3328f92a0c2749deefcd863.png

通过上面两点,任何时候,不同线程总能看到该变量的最新值.所有的操作都是CPU级别的。

并不是说使用了volatile就线程安全了

package com.keytech.task;

import java.util.concurrent.ExecutorService;

import java.util.concurrent.Executors;

import java.util.concurrent.Semaphore;

public class VolatileTest {

private static Integer clientTotal=5000;

private static Integer threadTotal=200;

private static volatile Integer count=0;

public static void main(String[] args) {

ExecutorService executorService = Executors.newCachedThreadPool();

Semaphore semaphore=new Semaphore(threadTotal);

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

executorService.execute(()->{

try{

semaphore.acquire();

update();

semaphore.release();

}catch (Exception e){

e.printStackTrace();

}

});

}

executorService.shutdown();

System.out.println("count:"+count);

}

private static void update(){

count++;

}

}

//count:4988

虽然使用了volatile,但是线程不安全。原因:update是非原子性的。

private static void update() {

count++; //分3步

//1.取出当前count值

//2.count + 1

//3.count 重新写回主存

}

假设同时有两个线程进行操作,两个线程同时执行到第一步(从内存中读取最新值)得到一样的最新的结果,然后进入第二步(+1操作)并进行第三步(从新写回主存)。尽管第一步获取的值是一样的,但是同时将+1后的操作写回主存,这样就会丢掉某个+1的操作,这样就会出现线程不安全问题

总结

volatile进行多线程加是线程不安全的,不适合计数

volatile不具备原子性

volatile的使用场景

对变量的写操作不依赖当前值

该变量没有包含在其它变量的不变式子中

volatile适合作为状态的标记量

volatile boolean flag = false;

//线程1

context = loadContext();

flag = true;

//线程2

while(!flag){

sleep();

}

todo(context);

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值