• 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()为何会影响内存可见性的原因了。

  • 相关阅读:
    数据挖掘竞赛常用代码段
    东南大学《算法设计基础》课程作业 4
    东南大学《算法设计基础》课程作业 3
    东南大学《算法设计基础》课程作业 2
    东南大学《算法设计基础》课程作业 1
    shiro
    你好2021!
    关系型数据库索引设计与优化
    lua table面向对象扩展
    lua、python对比学习
  • 原文地址:https://www.cnblogs.com/ljl150/p/12484877.html
Copyright © 2020-2023  润新知