先上代码,第一种情况,想在多线程运行的过程中,打出当前的线程名称,代码如下:
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,所以就跳出了循环。
本文通过两个实例探讨了`System.out.println`在多线程环境中的影响。在特定情况下,删除`println`会导致线程陷入无限循环。原因在于`println`方法的同步特性,它会强制线程获取锁、刷新内存,从而影响其他线程对共享变量的观察。
3691

被折叠的 条评论
为什么被折叠?



