JAVA多线程—共享变量的可见性问题

一、简介

我们知道线程在工作的时候有自己的私有内存,工作内存。程序运行的时候从主内存拉取需要的变量到工作内存,处理完再返回主内存。这篇文章总结哪些代码会使线程去主内存拉取变量。

二、volatile

volatile修饰的变量,不论什么语句都会从主内存拉取变量。
JAVA多线程—共享变量的可见性问题_第1张图片

该程序能顺利完成,不会死循环。因为“isStop ”用volatile修饰了

三、synchronized

调用其他synchronized加锁的代码块,系统会从主内存拉取变量。
JAVA多线程—共享变量的可见性问题_第2张图片
这种情况同样不会死循环。
注意,synchronized加锁后用到的变量才会从主内存拉取。

四、不会从主内存拉取的操作

只要不影响到变量的值,变量不参与运算,则不会从主内存中拉取,比如while,if,赋值给其他变量,作为方法参数等,如下图代码:
JAVA多线程—共享变量的可见性问题_第3张图片
这种情况,程序会死循环。它获得的isStop永远都是false。

如果调用了被synchronized加锁的方法,或者方法中有被synchronized加锁的代码块,那么变量将会从主内存中拉取。即使锁的是毫不相干的对象,同样会使得变量从主内存拉取,如下图这里的System.out.println锁的是PrintStream的实例。
JAVA多线程—共享变量的可见性问题_第4张图片
JAVA多线程—共享变量的可见性问题_第5张图片

五、从主内存中拉取的操作

变量被赋值,变量参与计算,比如加减乘除,字符串拼接,++,–等操作,线程休眠会使从主内存中拉取变量。下图这些情况都不会出现死循环:
JAVA多线程—共享变量的可见性问题_第6张图片JAVA多线程—共享变量的可见性问题_第7张图片

你可能感兴趣的:(JAVA多线程—共享变量的可见性问题)