了解Java内存模型(Java Memory Model, JMM)

2024-06-09 08:44

本文主要是介绍了解Java内存模型(Java Memory Model, JMM),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

了解Java内存模型(Java Memory Model, JMM)

Java内存模型(Java Memory Model, JMM)是Java语言规范中规定的一组规则,定义了多线程程序中变量(包括实例字段、静态字段和数组元素)的访问方式。JMM的设计目标是保证线程之间的内存可见性和操作的有序性,从而帮助开发者编写并发安全的程序。本文将详细介绍Java内存模型的各个方面。

1. 内存模型的基础

JMM定义了线程和主内存之间的抽象关系。在JMM中,每个线程都有一个私有的本地内存,包含了该线程使用到的变量的副本。线程之间的通信(例如一个线程写入一个变量,另一个线程读取这个变量)必须通过主内存来完成。一个线程对变量的修改必须在写入主内存后,其他线程才能看到这次修改。

2. 内存可见性

内存可见性是指一个线程对变量的修改,另一个线程能够看到的条件。JMM通过以下几个关键字来保证内存可见性:

2.1 volatile关键字

volatile关键字保证了变量在多个线程之间的可见性。当一个变量被声明为volatile时,JMM会保证所有线程都能看到这个变量的最新值。具体来说,JMM通过以下两个规则实现volatile的语义:

  • 当写入一个volatile变量时,JMM会强制将该变量的值刷新到主内存中。
  • 当读取一个volatile变量时,JMM会强制从主内存中读取该变量的最新值。

2.2 synchronized关键字

synchronized关键字用于实现线程之间的互斥访问。除了互斥功能外,synchronized还可以保证内存可见性:

  • 当一个线程进入synchronized块时,它必须先获得该块对象的锁,并清空本地内存中的变量副本。
  • 当一个线程退出synchronized块时,它必须将对变量的修改刷新到主内存中,并释放锁。

通过这种机制,synchronized保证了在synchronized块内对变量的修改对其他线程可见。

3. 有序性

有序性是指在多线程程序中,程序的执行顺序与代码的顺序一致。JMM通过以下几个方面来保证有序性:

3.1 happens-before规则

JMM通过happens-before原则来定义操作之间的有序性。happens-before是指如果一个操作happens-before另一个操作,那么第一个操作的结果对第二个操作可见,并且第一个操作在第二个操作之前执行。JMM定义了以下几种常见的happens-before关系:

  • 程序次序规则:在一个线程内,按照代码顺序,前面的操作happens-before后面的操作。
  • 锁定规则:一个unlock操作happens-before后面对同一个锁的lock操作。
  • volatile变量规则:对一个volatile变量的写操作happens-before后面对该变量的读操作。
  • 线程启动规则:Thread对象的start()方法happens-before该线程的每一个动作。
  • 线程终止规则:线程中的所有操作happens-before其他线程检测到该线程已经终止(通过Thread.join()或Thread.isAlive()方法)。

3.2 指令重排序

为了提高性能,编译器和处理器可能会对指令进行重排序,但这种重排序不会违反happens-before规则。JMM通过内存屏障(Memory Barriers)来禁止特定的重排序,从而保证程序的有序性和正确性。

4. 常见的内存一致性错误

在多线程编程中,未能正确处理内存可见性和有序性可能会导致一些常见的内存一致性错误,例如:

  • 读脏数据:一个线程读取了另一个线程未同步的写入数据。
  • 重排序导致的数据丢失:指令重排序导致程序的执行结果与预期不一致。

为了避免这些问题,开发者应当遵循JMM规定的规则,正确使用volatile和synchronized等关键字。

参考链接

  1. Java内存模型(JMM)详解:https://docs.oracle.com/javase/specs/jls/se11/html/jls-17.html
  2. Java并发编程实践(Java Concurrency in Practice):https://jcip.net/

在这里插入图片描述

这篇关于了解Java内存模型(Java Memory Model, JMM)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

分布式锁在Spring Boot应用中的实现过程

《分布式锁在SpringBoot应用中的实现过程》文章介绍在SpringBoot中通过自定义Lock注解、LockAspect切面和RedisLockUtils工具类实现分布式锁,确保多实例并发操作... 目录Lock注解LockASPect切面RedisLockUtils工具类总结在现代微服务架构中,分布

Java使用Thumbnailator库实现图片处理与压缩功能

《Java使用Thumbnailator库实现图片处理与压缩功能》Thumbnailator是高性能Java图像处理库,支持缩放、旋转、水印添加、裁剪及格式转换,提供易用API和性能优化,适合Web应... 目录1. 图片处理库Thumbnailator介绍2. 基本和指定大小图片缩放功能2.1 图片缩放的

Spring Boot集成/输出/日志级别控制/持久化开发实践

《SpringBoot集成/输出/日志级别控制/持久化开发实践》SpringBoot默认集成Logback,支持灵活日志级别配置(INFO/DEBUG等),输出包含时间戳、级别、类名等信息,并可通过... 目录一、日志概述1.1、Spring Boot日志简介1.2、日志框架与默认配置1.3、日志的核心作用

破茧 JDBC:MyBatis 在 Spring Boot 中的轻量实践指南

《破茧JDBC:MyBatis在SpringBoot中的轻量实践指南》MyBatis是持久层框架,简化JDBC开发,通过接口+XML/注解实现数据访问,动态代理生成实现类,支持增删改查及参数... 目录一、什么是 MyBATis二、 MyBatis 入门2.1、创建项目2.2、配置数据库连接字符串2.3、入

Springboot项目启动失败提示找不到dao类的解决

《Springboot项目启动失败提示找不到dao类的解决》SpringBoot启动失败,因ProductServiceImpl未正确注入ProductDao,原因:Dao未注册为Bean,解决:在启... 目录错误描述原因解决方法总结***************************APPLICA编

深度解析Spring Security 中的 SecurityFilterChain核心功能

《深度解析SpringSecurity中的SecurityFilterChain核心功能》SecurityFilterChain通过组件化配置、类型安全路径匹配、多链协同三大特性,重构了Spri... 目录Spring Security 中的SecurityFilterChain深度解析一、Security

SpringBoot多环境配置数据读取方式

《SpringBoot多环境配置数据读取方式》SpringBoot通过环境隔离机制,支持properties/yaml/yml多格式配置,结合@Value、Environment和@Configura... 目录一、多环境配置的核心思路二、3种配置文件格式详解2.1 properties格式(传统格式)1.

Apache Ignite 与 Spring Boot 集成详细指南

《ApacheIgnite与SpringBoot集成详细指南》ApacheIgnite官方指南详解如何通过SpringBootStarter扩展实现自动配置,支持厚/轻客户端模式,简化Ign... 目录 一、背景:为什么需要这个集成? 二、两种集成方式(对应两种客户端模型) 三、方式一:自动配置 Thick

MySQL 内存使用率常用分析语句

《MySQL内存使用率常用分析语句》用户整理了MySQL内存占用过高的分析方法,涵盖操作系统层确认及数据库层bufferpool、内存模块差值、线程状态、performance_schema性能数据... 目录一、 OS层二、 DB层1. 全局情况2. 内存占js用详情最近连续遇到mysql内存占用过高导致

Spring WebClient从入门到精通

《SpringWebClient从入门到精通》本文详解SpringWebClient非阻塞响应式特性及优势,涵盖核心API、实战应用与性能优化,对比RestTemplate,为微服务通信提供高效解决... 目录一、WebClient 概述1.1 为什么选择 WebClient?1.2 WebClient 与