System.out.println对多线程的影响

本文通过两个实例探讨了`System.out.println`在多线程环境中的影响。在特定情况下,删除`println`会导致线程陷入无限循环。原因在于`println`方法的同步特性,它会强制线程获取锁、刷新内存,从而影响其他线程对共享变量的观察。

先上代码,第一种情况,想在多线程运行的过程中,打出当前的线程名称,代码如下:

public class VolatileVisibilityTest {
    private static boolean initFlag = false;

    public static void main(String[] args) throws InterruptedException {
        System.out.println("1 " + Thread.currentThread().getName());
        new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println("等待数据准备。。");
                while (!initFlag) {
                    System.out.println("2 " + Thread.currentThread().getName());
                }
                System.out.println("==================数据准备完毕,执行程序逻辑");
            }
        }).start();
        Thread.sleep(100);

        new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println("3 " + Thread.currentThread().getName());
                prepareData();
            }
        }).start();
    }

    private static void prepareData() {
        System.out.println("数据准备了ing。。。。。");
        initFlag = true;
        System.out.println("数据准备ok。");
    }

}

运行结果如下:

1 main
等待数据准备。。
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
2 Thread-0
3 Thread-1
2 Thread-0
数据准备了ing。。。。。
数据准备ok。
2 Thread-0
==================数据准备完毕,执行程序逻辑

Process finished with exit code 0

从运行结果可知,该线程Thread-0可以正常结束

 

第二种情况,将while循环里面的system.out.println进行注销,那么Thread-0就会一直陷入循环状态,运行结果如下:

1 main
等待数据准备。。
3 Thread-1
数据准备了ing。。。。。
数据准备ok。

比较纳闷,System.out.println居然会影响多线程运行结果,于是查找相关原因,查看println的源码如下:

public void println(String x) {
        synchronized (this) {
            print(x);
            newLine();
        }
    }

最终解释如下:

先解释在while循环里面将println注释后产生的结果原因:一开始主线程中的initFlag的变量值为false,创建了一个线程Thread-0,将initFlag复制到运行内存中,因为Thread-0在运行的时候initFlag一直都是false,因为while循环会一直运行,后面的线程Thread-1虽然改变了主内存里面initFlag为true了,但是影响不了Thread-0运行内存中的initFlag的值。因此Thread-0会一直在while中无限循环;

现在加了println后,因为println操作是Synchronized加锁的,它会做以下的操作:1.获得同步锁;2,清空工作流出来;3.从主内存拷贝对象副本到线程工作内存中;4.开始继续执行代码;5.刷新主内存数据;6.释放同步锁。

在清空内存刷新内存的过程中Thread-0线程的initFlag就变成了true,所以就跳出了循环。

 

评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值