编程 5 年之惑,AI 给出完美答案

前言

今年都还没写过一个需求呢😣,闲来无事,突发奇想 volatile 可见性的问题,我以前一直都没模拟成功过呢。现在AI这么厉害,应该能帮我解决这个困惑吧!!!

volatile变量的修改,能立即对其他线程可见(强制刷新主内存和本地缓存)。

相信所有的java🧔对这句话非常的熟悉了。有多少个人去做过测试,我3、5年前就已经去做过测试了。但是很遗憾,当时怎么也复现不了。不管我用不volatile 效果都一样呢。今天AI 帮我解决了我的困惑!!!

🎈不想看过程的,直接看总结吧!

可见性理论测试

错误的测试案例

下面是我做的测试代码,没有用volatile修饰,所以理论上来说一直会输出Worker thread running...

实现代码:

java 复制代码
public class Test {
    private static boolean stop = false;

    public static void main(String[] args) {
        Thread workerThread = new Thread(() -> {
            int count = 0;
            // 按照没有用volatile修饰,当前线程不能看到变量被修改,所以理论上来说不会退出循环
            while (!stop) {
                System.out.println("Worker thread running...");
                count++;
            }
            System.out.println("Worker thread stopped, count: " + count);
        });

        workerThread.start();

        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

        stop = true;
        System.out.println("Main thread set stop to true");
    }

}

执行结果

text 复制代码
Worker thread running...
Main thread set stop to true
Worker thread stopped, count: 1

竟然和想象中的完全不一样。

有大佬看到这儿已经定位到了问题了么❔

AI解惑

为什么我不用volatile修饰 其他线程也能看到变化呢

就是说在某些情况下会去与主存同步,获取最新值。下面列举了比如CPU混存满了,或者到了特定的时间间隔。

如果是因为这两个原因,就有点说不通了,我就运行一个demo,而且每次执行都是同样的结果。

没有获取到我想要的答案,那就继续AI吧✈

能帮我举一个 未用volatile修饰变量,其他线程看不到变量变化的

  • 看一下AI是怎么解释的吧,如下图:*

运行上面的代码,确实出现 无法对出循环的问题。 这个例子和我开篇的例子就 少了一个System.out.println("Worker thread running...");

🤩看到这儿 ,应该❗大概❗好像❗感觉❗知道问题所在了吧❓

我的猜测

可能是System.out.println()加了锁的,导致线程的工作内存被刷新了。 点进去看果然sout(打印方法简称)方法是加了锁的。

java 复制代码
public void println(String x) {
    if (getClass() == PrintStream.class) {
        //也加了锁的
        writeln(String.valueOf(x));
    } else {
        synchronized (this) {
            print(x);
            newLine();
        }
    }
}

继续问AI,看看AI是怎么解释的

为什么用了sout,循环就会退出(没有用volatile,其他线程也能看到修改)

看一下AI是怎么解释的吧,如下图:

测试结论

在lock、sync修饰的代码,工作内存的变量会被刷新。所有我们在 打印 没有用volatile 修饰的变量的时候,变量也会被刷新,导致测试结果偏离预期

Tips:用在while 循环中使用Thread.sleep() 也会退出循环,是不是也意味着它也能刷线程的工作内存呢?

目前作者也是没有找到比较官方的资料

总结

本篇文章利用AI 分析了 无法复现普通变量 的不可见性 的原因。最后确认是因为使用sout访问变量,sout底层加锁,所以变量在工作内存中被刷新了,导致程序员的运行结果和理论上的不一致。

原来小小sout竟然有这么大的能量😱,兄弟们好好的把日志打印给我用起来,防御式编程✈啊,让后面的同学删除一行日志,就导致运行结果就变了,这不美滋滋。

最后,想了解volatile更多的知识的 ,推荐阅读:深入解析Java中volatile关键字的底层原理

相关推荐
一只乔哇噻3 分钟前
java后端工程师进修ing(研一版 || day41)
java·开发语言·学习·算法
User_芊芊君子21 分钟前
【Java】设计模式——单例、工厂、代理模式
java·设计模式·代理模式
2301_8035545230 分钟前
正向代理,反向代理,负载均衡还有nginx
java·nginx·负载均衡
要开心吖ZSH31 分钟前
软件设计师备考-(十六)数据结构及算法应用(重要)
java·数据结构·算法·软考·软件设计师
向上的车轮39 分钟前
基于Java Spring Boot的云原生TodoList Demo 项目,验证云原生核心特性
java·spring boot·云原生
程序员清风41 分钟前
快手一面:为什么要求用Static来修饰ThreadLocal变量?
java·后端·面试
逍遥德42 分钟前
Java8 Comparator接口 和 List Steam 排序使用案例
java·spring boot·list·排序算法
前行的小黑炭1 小时前
Android :如何快速让布局适配手机和平板?
android·java·kotlin
_BugMan2 小时前
【IDEA】干活?一个IDEA即可,集成开发平台打造攻略
java·ide·intellij-idea
YA3333 小时前
java设计模式二、工厂
java·开发语言·设计模式