Java中System.out.println()为何会影响内存可见性?

我们先来看段代码:

 1 class ThreadVolatileDemo extends Thread{
 2     static boolean flag=true;//注意该变量没有被volatile修饰
 3     @Override
 4     public void run() {
 5         while(flag){
 6             System.out.println("子线程");
 7         }
 8         System.out.println("子线程结束");
 9     }
10 }
11 public class ThreadVolatile {
12     public static void main(String[] args) {
13         ThreadVolatileDemo tVD=new ThreadVolatileDemo();
14         tVD.start();
15         try {
16             Thread.sleep(1000);
17         } catch (InterruptedException e) {
18             e.printStackTrace();
19         }
20         System.out.println("一秒钟一结束");
21         ThreadVolatileDemo.flag=false;//flag变量在一秒后在主线程中被修改为false
22     }
23 }

  通过看代码我们可以知道这是一个简单的多线程代码,子线程的run方法也很简单,就是一个单纯的while循环,我们先思考一下这段代码可能的运行结果,看代码可知,flag是一个普通变量,初始值为true,且没有被volatile修饰,也就是说它不具备内存可见性,又因为主线程中修改flag变量是在一秒之后的,然而这时候子线程已经开启了,且已经拥有了自己的本地内存,里面也已经存储了flag变量的副本,因为没有被volatile修饰,不具有可见性,所以子线程就不再和主内存中该变量值有任何关系,而是直接操作在本地内存上的变量值。因此由于子线程开启后flag变量的副本值一直为true,所以子线程就一直陷入在while死循环中出不来。

  但是!!! 当我运行完之后却发现,代码运行的结果跟我预想的有很大的差别,下面是代码的真正运行结果。

  通过结果可知,代码并没有陷入到循环中,这是为什么呢???

  于是我稍稍改动了下子线程run方法中的代码,如下所示,改完之后我发现代码的运行结果跟我之前分析的结果却是一样的,代码陷入了死循环中,这样看来问题是出在了 【System.out.println("子线程");】这句上。难道说这条打印语句已经影响到了内存可见性吗?

 1 class ThreadVolatileDemo extends Thread{
 2     static boolean flag=true;//注意该变量没有被volatile修饰
 3     @Override
 4     public void run() {
 5         while(flag){
 6             //System.out.println("子线程");
 7             //屏蔽掉while循环中的打印语句
 8         }
 9         System.out.println("子线程结束");
10     }
11 }

改完之后代码的运行结果:

 通过查看println源码,可以发现println语句中有一个上锁的操作:

  

通过查资料发现,在使用了synchronized上锁这个操作后线程会做以下操作:

  1.获得同步锁

  2.清空工作内存

  3.从主内存中拷贝对象副本到本地内存

  4.执行代码(打印语句或加加操作)

  5.刷新主内存数据

  6.释放同步锁

这也就是System.out.println()为何会影响内存可见性的原因了。

 

posted @ 2020-03-13 10:53  小L要努力吖  阅读(1150)  评论(0编辑  收藏  举报