volatile是否线程安全的
证明程序:
public class TestVolatile { public static volatile int numb = 0; public static void main(String[] args) throws Exception { for (int i = 0; i < 100; i++) { new Thread(i+""){ @Override public void run() { for (int i = 0; i < 1000; i++) { numb=numb+1; } } }.start(); } Thread.sleep(2000); System.out.println(numb); } }
正确结果:100000
事实是:
90000多,不固定
事实胜于雄辩,所以volatile不是线程安全的
原因:
jvm有一个主内存,volatile修饰的变量还是在主内存里面,而线程去对变量进行+运算,是在线程的工作内存修改的,不能立刻及时同步到主内存,多线程情况下,还是会存在看到变量不一样的情况 ,volatile修饰的变量,在有多线程对变量做运算并不安全。
专业解释:
修改volatile变量分为四步:
1)读取volatile变量到local
2)修改变量值
3)local值写回
4)插入内存屏障,即lock指令,让其他线程可见
这样就很容易看出来,前三步都是不安全的,取值和写回之间,不能保证没有其他线程修改。原子性需要锁来保证。