JVM之Java内存模型

Java内存模型(Java Memory Model,简称JMM)是Java虚拟机(JVM)规范中定义的一套规则,用于描述多线程环境下变量如何被访问和同步。在多线程编程中,内存模型的重要性不言而喻,它直接关系到程序的正确性和性能。本文将深入探讨Java内存模型的原理、特性、示例以及总结。

一、基本概念

1. 主内存与工作内存

  • 主内存:主内存是JVM中所有线程共享的内存区域,用于存储Java程序中的变量和对象实例等数据。所有的共享变量都存储在主内存中,包括实例字段、静态字段和数组元素等。
  • 工作内存:每个线程都有自己的工作内存,也称为本地内存。工作内存是线程私有的,用于存储主内存中共享变量的副本。线程对变量的所有操作(读取、写入)都在工作内存中进行,而不是直接操作主内存。

2. 内存交互操作

JMM定义了以下八种操作来完成主内存和工作内存之间的交互:

  • lock(锁定):作用于主内存的变量,将变量标识为一条线程独占状态。
  • unlock(解锁):作用于主内存的变量,释放处于锁定状态的变量,允许其他线程锁定该变量。
  • read(读取):作用于主内存的变量,将变量值从主内存传输到线程的工作内存中。
  • load(载入):作用于工作内存的变量,将read操作从主内存中得到的变量值放入工作内存的变量副本中。
  • use(使用):作用于工作内存的变量,将工作内存中的一个变量值传递给执行引擎。
  • assign(赋值):作用于工作内存的变量,将执行引擎接收到的值赋给工作内存的变量。
  • store(存储):作用于工作内存的变量,将工作内存中的一个变量的值传送到主内存中。
  • write(写入):作用于主内存的变量,将store操作从工作内存中得到的变量的值放入主内存的变量中。

三、特性

1. 原子性(Atomicity)

  • 定义:一个或多个操作,要么全部执行,要么全部不执行,在执行过程中不会被任何因素打断。
  • 实现方式:在Java中,对基本数据类型的访问和操作是原子的,例如对int、long、short、byte、char、boolean和reference类型的读写操作都是原子的。对于复合操作(如i++),则需要通过同步机制(如synchronized、Lock等)来保证原子性。

2. 可见性(Visibility)

  • 定义:一个线程对共享变量的修改,能够被其他线程看到。

  • 实现方式:Java内存模型通过volatile关键字和synchronized关键字来实现可见性。

    • volatile关键字:当一个变量被声明为volatile时,对该变量的读写操作都会直接作用于主内存,从而保证了可见性。但是,volatile关键字不能保证操作的原子性。
    • synchronized关键字:synchronized关键字通过锁定机制来确保只有一个线程可以执行一段代码。当一个线程访问一个对象的synchronized方法或代码块时,它将获得该对象的锁,其他线程则必须等待。在解锁之前对变量的修改对其他线程是可见的。

3. 有序性(Ordering)

  • 定义:程序执行的顺序按照代码的先后顺序执行。
  • 实现方式:Java内存模型通过Happens-Before原则来定义操作之间的偏序关系,从而允许一定程度的重排序,但同时又保证程序最终执行的结果与预期一致。

三、Happens-Before原则

Happens-Before原则是Java内存模型中定义的一组偏序关系,用于判断两个操作之间的内存可见性和有序性。它包括以下几种情况:

* 程序次序规则

一个线程中的每个操作,Happens-Before于该线程中的任意后续操作。

* 监视器锁规则

对一个锁的解锁,Happens-Before于随后对这个锁的加锁。

* volatile变量规则

对一个volatile变量的写,Happens-Before于任意后续对这个volatile变量的读。

* 传递性

如果A Happens-Before B,且B Happens-Before C,那么A Happens-Before C。

* 线程启动规则

Thread对象的start()方法调用Happens-Before于该线程的每一个动作。

* 线程终止规则

线程的所有操作都Happens-Before于其他线程检测到这个线程已经终止。

* 线程中断规则

对线程interrupt()方法的调用Happens-Before于被中断线程的代码检测到中断事件的发生。

* 对象终结规则

一个对象的初始化完成(构造函数执行结束)Happens-Before于它的finalize()方法的开始。

四、代码示例

1. 可见性示例

代码示例

public class VisibilityExample {private static boolean ready;private static int number;public static void main(String[] args) throws InterruptedException {Thread one = new Thread(() -> {number = 42;ready = true; // ①});Thread two = new Thread(() -> {while (!ready) {Thread.onSpinWait(); // ②}System.out.println(number); // ③});one.start();two.start();one.join();}
}

问题分析

  • 在这个例子中,线程one设置了number的值并标记ready为true。线程two在一个循环中等待ready变为true,然后打印number的值。
  • 如果没有正确的同步机制,线程two可能看不到线程one对number和ready的修改,因为这两个线程的工作在不同的内存区域中。这可能导致线程two打印出number的初始值(0),而不是线程one设置的值(42)。

解决方案

  • 可以使用volatile关键字或synchronized关键字来确保内存可见性。在这个例子中,如果将ready声明为volatile,则线程one对ready的修改将立即对线程two可见,从而确保线程two能够正确打印出number的值。

修改后的代码

public class VisibilityExample {private static volatile boolean ready; // 使用volatile关键字private static int number;public static void main(String[] args) throws InterruptedException {Thread one = new Thread(() -> {number = 42;ready = true; // ①});Thread two = new Thread(() -> {while (!ready) {Thread.onSpinWait(); // ②}System.out.println(number); // ③});one.start();two.start();one.join();}
}

2. 有序性示例

代码示例

public class Singleton {private static Singleton instance;public static Singleton getInstance() {if (instance == null) {synchronized (Singleton.class) {if (instance == null) {instance = new Singleton();}}}return instance;}
}

问题分析

  • 在这个例子中,Singleton类使用了双重检查锁定(Double-Checked Locking)模式来确保单例模式的安全性。

  • 然而,由于指令重排序的存在,这种实现方式在某些情况下可能会出现问题。具体来说,new Singleton()这个操作实际上会有以下三个步骤:

    1. 分配一块内存M。
    2. 在内存M上初始化Singleton对象。
    3. 将M的地址赋值给instance变量。
  • 如果发生指令重排序,顺序可能会变为:

    1. 分配一块内存M。
    2. 将M的地址赋值给instance变量。
    3. 在内存M上初始化Singleton对象。
  • 在这种情况下,如果线程A在执行到步骤2时发生线程切换,切换到线程B。线程B在执行getInstance()方法时,会发现instance不为null,因此直接返回instance。然而,此时的instance对象可能还没有被初始化,如果此时访问instance的成员变量就可能触发空指针异常。

解决方案

  • 可以通过在对象引用前添加volatile关键字来解决这个问题。volatile关键字可以禁止指令重排序,确保对象在初始化完成后再将引用赋值给变量。

修改后的代码

public class Singleton {private static volatile Singleton instance;public static Singleton getInstance() {if (instance == null) {synchronized (Singleton.class) {if (instance == null) {instance = new Singleton();}}}return instance;}
}

五、内存模型与硬件内存架构的关系

Java内存模型并不是对硬件内存架构的简单模拟,而是对其的一种抽象和简化。现代计算机为了高效运行,在CPU与内存之间设置了高速缓存(Cache)作为数据缓冲,这一机制在多线程环境下可能会引发缓存一致性问题。

为了解决这个问题,处理器需要遵循一些缓存一致性协议,如MSI、MESI(Modified, Exclusive, Shared, Invalid)等。这些协议确保了多个处理器核心在访问共享内存时,能够保持数据的一致性。例如,在MESI协议中,缓存行(Cache Line)有四种状态:修改(Modified)、独占(Exclusive)、共享(Shared)和无效(Invalid)。通过这些状态,处理器可以跟踪缓存行的数据是否与其他核心保持一致。

Java内存模型(JMM)在设计时,考虑了这些硬件层面的缓存一致性问题。JMM定义了一套规范,用于描述变量在内存中的存储方式、线程如何读取和写入内存中的变量,以及这些操作如何与其他线程进行同步。JMM的核心目标是确保在多线程环境中,程序的执行结果具有可预测性,即程序的行为与单线程环境下的行为一致(在遵守JMM规则的前提下)。

为了实现这一目标,JMM引入了几个关键概念:

1. 主内存与工作内存

JMM将内存分为两部分:主内存(Main Memory)和工作内存(Working Memory)。主内存是所有线程共享的,它存储了程序中的所有变量。每个线程都有自己的工作内存,它是线程私有的,存储了线程从主内存中读取的变量的副本。线程对变量的所有读写操作都必须在工作内存中进行,不能直接操作主内存中的变量。

2. volatile关键字

volatile是Java中的一个关键字,用于修饰变量。被volatile修饰的变量具有可见性和有序性两个特性。可见性意味着当一个线程修改了volatile变量的值,其他线程会立即看到这个修改。有序性则限制了编译器和处理器对volatile变量相关操作的重排序,从而保证了程序执行的正确性。

3. 原子性、可见性和有序性

原子性是指操作是不可中断的,要么全部执行完,要么完全不执行。可见性是指一个线程对共享变量的修改对其他线程是可见的。有序性是指程序执行的顺序是按照代码的顺序来的(在遵守JMM规则的前提下)。JMM通过一系列规则来保证这三个特性的实现。

4. 先行发生原则(Happens-Before)

这是JMM定义的一种偏序关系,用于描述两个操作之间的执行顺序。如果操作A先行发生于操作B,那么操作A对共享变量的影响在操作B中是可见的。先行发生原则是判断数据是否存在竞争、线程是否安全的主要依据。

总结

Java内存模型(JMM)是JVM规范中定义的多线程环境下变量访问和同步的规则。它抽象了硬件内存架构,分为主内存和工作内存,线程通过工作内存与主内存交互。JMM保证了原子性、可见性和有序性,其中volatile和synchronized是关键实现方式。Happens-Before原则定义了操作间的偏序关系,确保内存可见性和有序性。示例展示了可见性和有序性问题及解决方案。JMM还考虑了硬件缓存一致性问题,通过规范确保多线程环境下程序执行的可预测性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/66304.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

安装教程:慧集通集成平台(DataLinkX)智能体客户端安装操作(Linux/windows/mac)

1.下载客户端 使用提供的账号登录集成平台后台(https://www.datalinkx.cn/),点击左侧菜单栏【智能体】→【智能体】进入到智能体列表界面,在该界面我们找到功能栏中的下载按钮点击则会弹出下载界面,在该界面我们可以选择不同的系统操作系统来下载对应版…

067B-基于R语言平台Biomod2模型的物种分布建模与数据可视化-高阶课程【2025】

课程培训包含:发票全套软件脚本学习数据视频文件导师答疑 本教程旨在通过系统的培训学习,学员可以掌握Biomod2模型最新版本的使用方法,最新版包含12个模型(ANN, CTA, FDA, GAM, GBM, GLM, MARS, MAXENT, MAXNET, RF, SRE, XGBOOST…

贵州省贵安新区地图+全域数据arcgis格式shp数据矢量路网地名+卫星影像底图下载后内容测评

贵州省贵安新区地图全域数据arcgis格式shp数据矢量路网地名卫星影像底图 贵安新区地图是一款基于ArcGIS格式的地理信息系统数据集,包含2022年3月更新的详尽矢量路网、地名信息以及卫星影像底图。这款数据集是针对贵安新区这一特定区域设计的,对于规划、…

pycharm完成git项目的拉取及修改,并保存为自己的项目,以cosyvoice为例

1.新建一个项目,如图 等待几分钟。因为还要虚拟环境的建立等。 2.克隆项目源码及提前准备(备) a.git的路径。 b.githube相关设置 左角。查看。 在对话框中,填写要摘取的仓库,提示非空目录。我给加了一个main 完成后&a…

Appium(一)--- 环境搭建

一、Android自动化环境搭建 1、JDK 必须1.8及以上(1) 安装:默认安装(2) 环境变量配置新建JAVA_HOME:安装路径新建CLASSPath%JAVA_HOME%\lib\dt.jar;%JAVA_HOME%\lib\tools.jar在path中增加:%JAVA_HOME%\bin;%JAVA_HOME%\jre\bin;(3) 验证…

Ant Design中Flex布局、Grid布局和Layout布局详解

好的,我们来更详细地探讨 Ant Design 中的 Flex布局、Grid布局 和 Layout布局 的特点、用法、适用场景,以及如何灵活运用它们来构建页面。下面将从各个方面进行更深入的分析,并提供具体的实例。 VueFlex布局实现响应式布局 1. Flex布局 概念…

Redis两种主要的持久化方式是什么?

Redis支持两种主要的持久化方式,它们分别是RDB(Redis Database Snapshotting)和AOF(Append Only File)。以下是这两种持久化方式的详细介绍: 一、RDB(Redis Database Snapshotting) …

flink cdc oceanbase(binlog模式)

接上文:一文说清flink从编码到部署上线 环境:①操作系统:阿里龙蜥 7.9(平替CentOS7.9);②CPU:x86;③用户:root。 预研初衷:现在很多项目有国产化的要求&#…

C++文件流 例题

问题: 设计一个留言类,实现以下的功能: 1) 程序第一次运行时,建立一个 message.txt 文本文件,并把用户输入的信息存入该文件; 2) 以后每次运行时,都先读取该文件的内容并显示给用户&#xff…

遮挡半透明效果

1、遮挡半透明效果是什么 在游戏开发中,遮挡半透明效果就是物体被挡住的部分,也能呈现出一种半透明效果而被看到(具体效果可以自定义)比如 当角色在建筑物之间穿行时,被遮挡部分能够呈现出半透明效果而被我们看到。遮…

大模型高效推理综述

大模型高效推理综述 1 Introduction2 Preliminaries2.1 transformer架构的LLM2.2 大模型推理过程2.3 推理效率分析 3 TAXONOMY(分类)4.数据级别优化4.1输入压缩4.1.1 提示词裁剪(prompt pruning)4.1.2 提示词总结(prompt summary)…

计算机网络--UDP和TCP课后习题

【5-05】 试举例说明有些应用程序愿意采用不可靠的UDP, 而不愿意采用可靠的TCP。 解答: 这可能有以下几种情况。 首先,在互联网上传输实时数据的分组时,有可能会出现差错甚至丢失。如果利用 TCP 协议对这些出错或丢失的分组进行重传&…

Go语言的基础知识

1, Go 语言介绍 Go 即 Golang,是 Google公司2009年11月正式对外公开的一门编程语言。 根据 Go 语言开发者自述,近10多年,从单机时代的C语言到现在互联网时代的Java,都没有令人满意的开发语言,而C往往给人的感觉是,花了100%的经历…

【UE5 C++课程系列笔记】20——共享指针的简单使用

目录 概念 创建共享指针示例 重设共享指针 共享指针内容转移 共享指针和共享引用的转换 判断共享指针的相等性 共享指针访问成员函数 自定义删除器 概念 共享指针(主要以 TSharedPtr 为例),TSharedPtr 基于引用计数机制来工作&#x…

flux中的缓存

1. cache,onBackpressureBuffer。都是缓存。cache可以将hot流的数据缓存起来。onBackpressureBuffer也是缓存,但是当下游消费者的处理速度比上游生产者慢时,上游生产的数据会被暂时存储在缓冲区中,防止丢失。 2. Flux.range 默认…

Ubuntu网络连接问题(笔记本更换wifi后,虚拟机连不上网络)

1、笔记本更换wifi后,虚拟机的IP地址变了,然后就连不上网络了(主机笔记本连接wifi正常上网) 2、修改子网地址(按照ubutun的ip设置子网掩码) 3、Ubuntu已经显示网络连接正常了,但是就是无法上网&…

如何在 Ubuntu 22.04 上安装 Cassandra NoSQL 数据库教程

简介 本教程将向你介绍如何在 Ubuntu 22.04 上安装 Cassandra NoSQL 数据库。 Apache Cassandra 是一个分布式的 NoSQL 数据库,旨在处理跨多个普通服务器的大量数据,并提供高可用性,没有单点故障。Apache Cassandra 是一个高度可扩展的分布…

Spring MVC实战指南:构建高效Web应用的架构与技巧(三)

响应数据和结果视图(7种) 返回值分类 创建web.xml&#xff08;spring、过滤器解决乱码、配置控制器dispatcherServlet、加载springmvc.xml文件、配置启动加载&#xff09;创建springmvc.xml文件 <!--配置了内容&#xff0c;启动Tomcat服务器的时候&#xff0c;就会被加载--…

oscp备考 oscp系列——Kioptix Level 1靶场 古老的 Apache Vuln

目录 前言 1. 主机发现 2. 端口扫描 3. 指纹识别 4. 目录扫描 5. 漏洞搜索和利用 前言 oscp备考&#xff0c;oscp系列——Kioptix Level 1靶场 Kioptix Level 1难度为简单靶场&#xff0c;主要考察 nmap的使用已经是否会看输出&#xff0c;以及是否会通过应用查找对应漏…

Linux下编译安装PETSc

本文记录在Linux下编译安装PETSc的流程。 零、环境 操作系统Ubuntu 22.04.4 LTSVS Code1.92.1Git2.34.1GCC11.4.0CMake3.22.1oneAPI2024.2.1 一、安装依赖 1.1 安装oneAPI 参见&#xff1a;Get the Intel oneAPI Base Toolkit , Get the Intel oneAPI HPC Toolkit 1.2 安…