深入理解 Java 内存模型(Java Memory Model, JMM)

2024-05-26 04:04

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

深入理解 Java 内存模型(Java Memory Model, JMM)

Java 内存模型(Java Memory Model, JMM)是 Java 并发编程的基础,规定了多线程环境中变量的访问和修改行为。为了更好地理解 JMM,需要了解它如何与系统内核和 CPU 交互,尤其是涉及 CPU 的缓存机制、缓存一致性协议和内存屏障等方面。

1. JMM 的基本概念

JMM 解决了两个核心问题:可见性 和 有序性。

  • 可见性:一个线程对共享变量的修改何时对其他线程可见。
  • 有序性:程序执行的顺序是否符合代码编写的逻辑顺序。

CPU 和系统内核通过缓存一致性协议和内存屏障来实现这些特性。

2. CPU 缓存与可见性

CPU 为了提高性能,会在处理器核心中使用缓存存储数据。每个核心都有自己的缓存(如 L1、L2 和 L3 缓存),线程对变量的操作首先会在缓存中进行,然后再写回主内存。不同线程运行在不同的处理器核心上时,对共享变量的修改可能不会立即对其他线程可见。

缓存一致性协议(如 MESI 协议)用于确保多个处理器核心的缓存数据一致。MESI 协议中的四个状态分别是:修改(Modified)、独占(Exclusive)、共享(Shared)和无效(Invalid)。当一个核心修改缓存中的数据时,其他核心会被通知数据已失效,需要从主内存中重新读取。

3. 内存屏障与有序性

内存屏障(Memory Barriers)是一种 CPU 指令,用于防止处理器对特定操作进行重排序,从而保证指令执行的顺序。内存屏障在 JMM 中起到了关键作用,确保变量的可见性和有序性。

内存屏障主要分为以下几种:

  • Load Barrier(加载屏障):禁止加载操作重排序。
  • Store Barrier(存储屏障):禁止存储操作重排序。
  • Full Barrier(全屏障):禁止所有类型的重排序。

volatile 关键字在 Java 中使用内存屏障来确保对变量的读写操作不会被重排序,并且修改立即对其他线程可见。

class SharedData {private volatile boolean flag = false;public void setFlag(boolean flag) {this.flag = flag;}public boolean isFlag() {return flag;}
}

在这个示例中,flag 变量被声明为 volatile,确保每次对 flag 的修改立即刷新到主内存,其他线程能及时看到修改。

4. 指令重排序与有序性

指令重排序(Instruction Reordering)是指编译器和处理器为优化性能而对指令执行顺序进行调整。为了保证多线程程序的正确性,JMM 通过内存屏障和 happens-before 规则来限制重排序。

Happens-Before 规则

  • 程序次序规则:在一个线程内,按照代码顺序,前面的操作 happens-before 后面的操作。
  • 监视器锁规则:一个锁的解锁操作 happens-before 后续的加锁操作。
  • volatile 变量规则:对一个 volatile 变量的写操作 happens-before 后续对该变量的读操作。
  • 传递性规则:如果 A happens-before B,且 B happens-before C,则 A happens-before C。
  • 线程启动规则:Thread.start() 方法调用 happens-before 启动线程中的任何操作。
  • 线程终止规则:线程中的所有操作 happens-before 其他线程检测到该线程终止。
  • 线程中断规则:对线程的中断操作 happens-before 被中断线程检测到中断事件。

5. JMM 的实现与系统内核和 CPU

JMM 通过内存屏障和缓存一致性协议在系统内核和 CPU 层面实现。

  • 内存屏障:用于强制在特定点刷新 CPU 缓存,确保指令的执行顺序。例如,volatile 关键字在底层实现中使用内存屏障,防止对 volatile 变量的访问被重排序。
class VolatileExample {private volatile int value;public void writer() {value = 1;  // 写操作}public int reader() {return value;  // 读操作}
}

在这个示例中,writer() 方法中的写操作和 reader() 方法中的读操作通过内存屏障实现可见性和有序性。

  • 缓存一致性协议:如 MESI 协议(修改、独占、共享、无效),确保多个处理器核心的缓存数据一致。每当一个核心修改缓存中的数据时,其他核心会被通知数据已失效,需从主内存中重新读取。

6. 同步机制

为了避免线程安全问题,Java 提供了多种同步机制来协调线程对共享变量的访问。

6.1 Synchronized

synchronized 关键字用于对代码块或方法进行加锁,确保同一时刻只有一个线程可以执行被加锁的代码。

class Counter {private int count = 0;public synchronized void increment() {count++;}public synchronized int getCount() {return count;}
}

在上述示例中,synchronized 确保了 increment 和 getCount 方法在多线程环境下的安全性。

6.2 Lock

Lock 接口提供了更灵活的锁机制,可以显式地加锁和解锁。

import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;class Counter {private int count = 0;private final Lock lock = new ReentrantLock();public void increment() {lock.lock();try {count++;} finally {lock.unlock();}}public int getCount() {lock.lock();try {return count;} finally {lock.unlock();}}
}

6.3 Volatile

volatile 关键字用于标记变量,使其对所有线程可见,禁止指令重排序。

class SharedData {private volatile boolean flag = false;public void setFlag(boolean flag) {this.flag = flag;}public boolean getFlag() {return flag;}
}

6.4 并发容器

Java 提供了一些线程安全的并发容器,简化了多线程编程中的共享数据管理。

  • ConcurrentHashMap
  • CopyOnWriteArrayList
  • BlockingQueue
import java.util.concurrent.ConcurrentHashMap;class ConcurrentExample {private ConcurrentHashMap<String, Integer> map = new ConcurrentHashMap<>();public void add(String key, int value) {map.put(key, value);}public int get(String key) {return map.get(key);}
}

这些容器在内部使用了复杂的同步机制,确保在高并发环境下的线程安全和高效性。

总结

Java 内存模型(JMM)通过内存屏障、缓存一致性协议等机制在系统内核和 CPU 层面上实现,确保多线程程序的可见性和有序性。理解 JMM 及其底层实现,对于编写高效且正确的并发程序至关重要。通过合理使用 volatile、synchronized 以及并发工具类,开发者可以有效地解决多线程环境中的各种问题,确保程序在高并发环境下的正确性和性能。

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



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

相关文章

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Spring Security--Architecture Overview

1 核心组件 这一节主要介绍一些在Spring Security中常见且核心的Java类,它们之间的依赖,构建起了整个框架。想要理解整个架构,最起码得对这些类眼熟。 1.1 SecurityContextHolder SecurityContextHolder用于存储安全上下文(security context)的信息。当前操作的用户是谁,该用户是否已经被认证,他拥有哪些角色权限…这些都被保

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

大模型研发全揭秘:客服工单数据标注的完整攻略

在人工智能(AI)领域,数据标注是模型训练过程中至关重要的一步。无论你是新手还是有经验的从业者,掌握数据标注的技术细节和常见问题的解决方案都能为你的AI项目增添不少价值。在电信运营商的客服系统中,工单数据是客户问题和解决方案的重要记录。通过对这些工单数据进行有效标注,不仅能够帮助提升客服自动化系统的智能化水平,还能优化客户服务流程,提高客户满意度。本文将详细介绍如何在电信运营商客服工单的背景下进行

NameNode内存生产配置

Hadoop2.x 系列,配置 NameNode 内存 NameNode 内存默认 2000m ,如果服务器内存 4G , NameNode 内存可以配置 3g 。在 hadoop-env.sh 文件中配置如下。 HADOOP_NAMENODE_OPTS=-Xmx3072m Hadoop3.x 系列,配置 Nam

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06