当前位置: 首页 > news >正文

线程安全问题(内存可见性)

导致的原因

        内存可见性问题的出现主要是因为编译器优化多线程导致的

示例代码

package 线程安全问题;import java.util.Scanner;/*** Created with IntelliJ IDEA.* Description:* User: wuyulin* Date: 2023-07-26* Time: 13:49*/
public class Demo2 {private volatile static int isQuit=0;public static void main(String[] args) {Thread A=new Thread(()->{while (isQuit==0){}});Thread B=new Thread(()->{Scanner sc=new Scanner(System.in);System.out.print("isQuit=");isQuit=sc.nextInt();});A.start();B.start();}
}

出现的问题

        在上述代码中如果isQuit成员变量不加上volatile关键字的话,该程序就会出现bug,即使在B线程中修改了isQuit的值,在A线程中依然不会退出循环

出现问题的原因

        出现该问题的原因是编译器对我们的代码进行了优化

         因为isQuit==0分为两步操作,1.将isQuit从内存中读取到寄存器中,2.在寄存器中进行isQuit与0的比较
    而从内存中读取数据到寄存器中要消耗很长的时间,在寄存器中进行isQuit与0的比较消耗的时间很短
    编译器发现经过了很多次的循环isQuit的值都没有改变,于是做了一个大胆的决定,将从内存中读取isQuit这个操作优化掉,采用读取寄存器中的值,效率便得到了极大的提高
    但是此时在B线程中修改isQuit在内存中的值,A线程由于已经不在内存中读取isQuit的值了,所以A线程的循环不会受到影响,就导致了无法停止A线程的循环

 
解决的方法

        解决的方法:在isQuit成员变量声明时加上volatile(易变的)关键字,表示该成员变量是易变的,此时编译器就不会优化掉去内存中读取isQuit值的操作,这样B线程改变了内存中isQuit的值,A线程便能够读取到,结束循环

http://www.lryc.cn/news/96273.html

相关文章:

  • STM32MX配置EEPROM(AT24C02)------保姆级教程
  • 微信小程序 样式和全局配置
  • 一.初识C语言
  • filebeat到kafka示例
  • AlmaLinux系统下的Zabbix汉化
  • 【网络编程】(TCP流套接字编程 ServerSocket API Socket API 手写TCP版本的回显服务器 TCP中的长短连接)
  • 企业级PaaS低代码快开平台源码,基于 Salesforce Platform 的开源替代方案
  • 【LeetCode】72.编辑距离
  • 大模型,开源干不掉闭源
  • Redis 九种数据类型的基本操作
  • 爬取微博热搜榜并进行数据分析
  • 基于深度神经网络的肺炎检测系统实现
  • C# LINQ和Lambda表达式对照
  • 二、SQL-6.DCL-1).用户管理
  • ElasticSearch学习--数据聚合
  • PostMan+Jmeter工具介绍及安装
  • AutoSAR系列讲解(实践篇)7.4-实验:配置SWCRTE
  • 腾讯云内存型CVM服务器MA3、M6、M6ce和M5处理器CPU说明
  • 集睿致远推出CS5466多功能拓展坞方案:支持DP1.4、HDMI2.1视频8K输出
  • SQL中为何时常见到 where 1=1?
  • React AntDesign表批量操作时的selectedRowKeys回显选中
  • anydesk远程控制,主动连接。
  • Spring Data Redis操作Redis
  • sqlite触发器1
  • python中——requests爬虫【中文乱码】的3种解决方法
  • E. Nastya and Potions(DFS+记忆化搜索)
  • 什么是tcp rst以及什么时候产生?
  • Visual Studio Code配置免密远程开发环境
  • flutter android Webview 打开网页错误ERR_CLEARTEXT_NOT_PERMITTED 、 net:ERR_CACHE_MISS
  • ARP协议(地址解析协议)