【多线程】volatile关键字探究和System.out.println()隐式地插入内存屏障问题

本文主要是介绍【多线程】volatile关键字探究和System.out.println()隐式地插入内存屏障问题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

已知volatile 通过内存屏障保证有序性和可见性且能防止指令重排序

在验证volatile功能的过程中发现,以下代码的执行结果中包含了 分支线程的方法成功停止

public class TestCall implements Runnable{private static  boolean flag = true;@Overridepublic void run() {System.out.println("分支线程开始");int j = 0;while (flag){System.out.println("1");}System.out.println("分支线程的方法成功停止");}public static void main(String[] args) throws InterruptedException {new Thread(new TestCall()).start();Thread.sleep(2000);for (int i = 0; i < 10000; i++) {System.out.println("main"+i);if (i == 9090){System.out.println("停止方法启动");flag = false;System.out.println("分支线程开始停止!!");}}}
}

测试过程中偶然发现while中加入 System.out.println(“1”); 会使 while (flag)flag的值监测到主内存的修改。但是如果不加System.out.println(“1”); **System.out.println(“分支线程的方法成功停止”);**这句就不会输出。经过测试,得出结果:

System.out.println() 方法在 Java 中通常不会显式地引入内存屏障(memory barrier)。在某些情况下,JVM 为了保证 System.out.println() 的输出顺序正确,可能会隐式地插入内存屏障来确保指令重排不会影响到输出的顺序。

同时内存屏障是一种硬件机制,用来阻止编译器或处理器对内存操作进行重排序,从而确保内存操作按照特定的顺序执行。这对于多线程编程中的内存可见性非常重要。
尽管 System.out.println() 本质上是一个 I/O 操作,它主要关注的是将数据输出到标准输出流,但在 JVM 的实现中,为了保证输出的有序性和线程安全性,可能采取了一些措施,比如:
使用锁来确保多个线程不会同时调用 System.out.println(),防止输出混乱。
在调用前后可能隐式地插入内存屏障,以防止其他内存操作被重排序到 System.out.println() 调用之前或之后,从而保证输出的一致性。
需要注意的是,具体的实现细节依赖于 JVM 的版本和实现方式,因此不能一概而论。但是,为了保证输出的一致性和线程安全性,System.out.println() 可能会间接导致内存屏障的使用

这篇关于【多线程】volatile关键字探究和System.out.println()隐式地插入内存屏障问题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1117416

相关文章

Spring的RedisTemplate的json反序列泛型丢失问题解决

《Spring的RedisTemplate的json反序列泛型丢失问题解决》本文主要介绍了SpringRedisTemplate中使用JSON序列化时泛型信息丢失的问题及其提出三种解决方案,可以根据性... 目录背景解决方案方案一方案二方案三总结背景在使用RedisTemplate操作redis时我们针对

Kotlin Map映射转换问题小结

《KotlinMap映射转换问题小结》文章介绍了Kotlin集合转换的多种方法,包括map(一对一转换)、mapIndexed(带索引)、mapNotNull(过滤null)、mapKeys/map... 目录Kotlin 集合转换:map、mapIndexed、mapNotNull、mapKeys、map

nginx中端口无权限的问题解决

《nginx中端口无权限的问题解决》当Nginx日志报错bind()to80failed(13:Permissiondenied)时,这通常是由于权限不足导致Nginx无法绑定到80端口,下面就来... 目录一、问题原因分析二、解决方案1. 以 root 权限运行 Nginx(不推荐)2. 为 Nginx

解决1093 - You can‘t specify target table报错问题及原因分析

《解决1093-Youcan‘tspecifytargettable报错问题及原因分析》MySQL1093错误因UPDATE/DELETE语句的FROM子句直接引用目标表或嵌套子查询导致,... 目录报js错原因分析具体原因解决办法方法一:使用临时表方法二:使用JOIN方法三:使用EXISTS示例总结报错原

Windows环境下解决Matplotlib中文字体显示问题的详细教程

《Windows环境下解决Matplotlib中文字体显示问题的详细教程》本文详细介绍了在Windows下解决Matplotlib中文显示问题的方法,包括安装字体、更新缓存、配置文件设置及编码調整,并... 目录引言问题分析解决方案详解1. 检查系统已安装字体2. 手动添加中文字体(以SimHei为例)步骤

SpringSecurity整合redission序列化问题小结(最新整理)

《SpringSecurity整合redission序列化问题小结(最新整理)》文章详解SpringSecurity整合Redisson时的序列化问题,指出需排除官方Jackson依赖,通过自定义反序... 目录1. 前言2. Redission配置2.1 RedissonProperties2.2 Red

nginx 负载均衡配置及如何解决重复登录问题

《nginx负载均衡配置及如何解决重复登录问题》文章详解Nginx源码安装与Docker部署,介绍四层/七层代理区别及负载均衡策略,通过ip_hash解决重复登录问题,对nginx负载均衡配置及如何... 目录一:源码安装:1.配置编译参数2.编译3.编译安装 二,四层代理和七层代理区别1.二者混合使用举例

Javaee多线程之进程和线程之间的区别和联系(最新整理)

《Javaee多线程之进程和线程之间的区别和联系(最新整理)》进程是资源分配单位,线程是调度执行单位,共享资源更高效,创建线程五种方式:继承Thread、Runnable接口、匿名类、lambda,r... 目录进程和线程进程线程进程和线程的区别创建线程的五种写法继承Thread,重写run实现Runnab

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

Java内存分配与JVM参数详解(推荐)

《Java内存分配与JVM参数详解(推荐)》本文详解JVM内存结构与参数调整,涵盖堆分代、元空间、GC选择及优化策略,帮助开发者提升性能、避免内存泄漏,本文给大家介绍Java内存分配与JVM参数详解,... 目录引言JVM内存结构JVM参数概述堆内存分配年轻代与老年代调整堆内存大小调整年轻代与老年代比例元空