学习笔记04——JMM内存模型

news2025/2/25 14:05:51
一、Java内存模型(JMM)是什么?

Java内存模型(Java Memory Model, JMM)是Java多线程编程中共享内存的访问规则,定义了线程如何与主内存(Main Memory)和工作内存(Work Memory)交互,解决多线程并发中的原子性、可见性、有序性问题。确保多线程程序在不同架构的处理器和内存系统上都能正确运行。


二、核心概念
  1. 主内存与工作内存

    • 主内存:所有线程共享的内存区域,存储变量(实例字段、静态字段等),类似于计算机中的物理内存。

    • 工作内存:每个线程私有的内存区域,存储主内存变量的副本。

    • 交互规则:线程只能操作工作内存中的变量副本,修改后需同步回主内存。

  2. 原子性、可见性、有序性

    • 原子性:操作不可分割(如AtomicInteger),要不全部完成,要不全部不做。

      1. 基本数据类型的赋值操作(如int a = 10)。

      2. 使用synchronized关键字或Lock接口实现的同步代码块。

        public class Counter {
            private int count = 0;
        ​
            public synchronized void increment() {
                count++;
            }
        }
    • 可见性:一个线程对变量的修改能被其他线程立刻看到(volatilesynchronized)。

      1. volatile关键字:确保变量的修改立即刷新到主内存,其他线程读取时能获取最新值。

      2. synchronized关键字:线程退出同步块时,会将工作内存中的变量写回主内存。

        public class VolatileExample {
            private volatile boolean running = true;
        ​
            public void stop() {
                running = false;
            }
        ​
            public void run() {
                while (running) {
                    // do something
                }
            }
        }
    • 有序性:代码执行顺序与程序顺序一致(防止指令重排序)。

      1. happens-before原则:定义了操作的先后顺序。

      2. 内存屏障:防止指令重排序。

  3. Happens-Before原则 JMM定义的保证有序性的规则,若操作A happens-before 操作B,则A的结果对B可见。 常见规则:

    • 程序顺序规则 :在单线程中,代码的执行顺序符合书写顺序。

    • volatile变量规则

    • 锁规则(synchronizedLock

    • 线程启动/终止规则

    • 传递性规则


三、关键机制
  1. volatile关键字

    • 保证变量的可见性:直接读写主内存,禁止缓存。

    • 禁止指令重排序:通过插入内存屏障(Memory Barrier)。

    • 示例:双检锁单例模式中的volatile修饰实例对象。

  2. synchronized与锁

    • 通过监视器锁(Monitor)实现原子性和可见性。

    • 线程解锁前必须将变量同步回主内存。

  3. final关键字

    • 构造函数中对final域的写入,与后续引用赋值操作不会被重排序。

    • 确保其他线程看到final变量时,其初始化已完成。

  4. 并发工具类

    如CountDownLatch、CyclicBarrier等,用于更复杂的线程同步。


四、常见问题与面试考点
  1. volatile能否保证原子性?

    • 不能。volatile仅保证读写操作的原子性,但复合操作(如i++)仍需同步。

  2. DCL(双检锁)单例模式为什么要加volatile?

    • 防止指令重排序导致其他线程获取未初始化完成的对象。

  3. synchronized和ReentrantLock的区别?

    • synchronized是JVM内置锁,自动释放;ReentrantLock需手动加锁/解锁,支持公平锁、条件变量。

  4. 什么是内存可见性问题?如何解决?

    • 线程A修改变量后未及时同步到主内存,线程B读取旧值。

    • 解决:volatilesynchronizedLock

  5. 指令重排序的典型场景?

    • 单例模式初始化:new Object()可能被拆分为分配内存、初始化对象、赋值引用三步,重排序后导致空指针。

  6. 什么是“线程安全的发布”?

    • 对象在构造完成后才能被其他线程访问。

    • 方式:volatile、静态初始化块、final域、线程安全容器(如ConcurrentHashMap)。


jJ五、代码示例:内存可见性问题
public class VisibilityDemo {
    // 不加volatile可能导致死循环
    private static volatile boolean flag = true;
​
    public static void main(String[] args) throws InterruptedException {
        new Thread(() -> {
            while (flag) {} // 线程可能读取到旧的flag值
            System.out.println("Thread stopped.");
        }).start();
​
        Thread.sleep(1000);
        flag = false; // 主线程修改flag
    }
}

六、面试总结
  1. 必考知识点

    • volatile的作用与原理

    • synchronized的底层实现(Monitor、锁升级)

    • Happens-Before原则的规则

    • 指令重排序与内存屏障

  2. 加分回答

    • 结合JMM分析ConcurrentHashMap的线程安全设计。

    • 对比JMM与物理内存模型(CPU缓存、MESI协议)。

掌握JMM是写出高并发代码的基础,也是大厂面试的必考领域!

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

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

相关文章

Android 老项目 jcenter 库失效

最近重新维护了一些老项目发现大部分jcenter库失效了, Could not resolve com.xx:2.1.3. 如果你也遇到了,不妨试试 替换为 aliyun的jcenter服务,就不用一个个找代替库了。 project 下的 build.gradle 文件添加: maven { url htt…

2025.2.23机器学习笔记:PINN文献阅读

2025.2.23周报 一、文献阅读题目信息摘要Abstract创新点网络架构架构A架构B架构C 实验结论后续展望 一、文献阅读 题目信息 题目: Physics-Informed Neural Networks for Modeling Water Flows in a River Channel期刊: IEEE TRANSACTIONS ON ARTIFICI…

Python Django系列—入门实例(二)

数据库配置 现在,打开 mysite/settings.py 。这是个包含了 Django 项目设置的 Python 模块。 默认情况下,​ DATABASES 配置使用 SQLite。如果你是数据库新手,或者只是想尝试 Django,这是最简单的选择。SQLite 包含在 Python 中…

【DeepSeek系列】05 DeepSeek核心算法改进点总结

文章目录 一、DeepSeek概要二、4个重要改进点2.1 多头潜在注意力2.2 混合专家模型MoE2.3 多Token预测3.4 GRPO强化学习策略 三、2个重要思考3.1 大规模强化学习3.2 蒸馏方法:小模型也可以很强大 一、DeepSeek概要 2024年~2025年初,DeepSeek …

独立开发者之Google Analytics使用教程

Google Analytics(GA)是Google提供的一款免费的网络分析服务,用于追踪和报告网站流量。以下是独立开发者如何使用Google Analytics的详细教程: 1. 创建Google Analytics账户 注册Google账户:如果你还没有Google账户&…

C++ 编程语言简介

C 是一种通用编程语言,它是作为 C 语言的增强而开发的,以包含面向对象的范例。它是一种命令式和编译语言。 C 是一种高级的通用编程语言,专为系统和应用程序编程而设计。它由贝尔实验室的 Bjarne Stroustrup 于 1983 年开发,作为…

计算机毕业设计SpringBoot+Vue.js明星周边产品销售网站(源码+文档+PPT+讲解)

温馨提示:文末有 CSDN 平台官方提供的学长联系方式的名片! 温馨提示:文末有 CSDN 平台官方提供的学长联系方式的名片! 温馨提示:文末有 CSDN 平台官方提供的学长联系方式的名片! 作者简介:Java领…

使用Windbg调试目标进程排查C++软件异常的一般步骤与要点分享

目录 1、概述 2、将Windbg附加到已经启动起来的目标进程上,或者用Windbg启动目标程序 2.1、将Windbg附加到已经启动起来的目标进程上 2.2、用Windbg启动目标程序 2.3、Windbg关联到目标进程上会中断下来,输入g命令将该中断跳过去 3、分析实例说明 …

ddd 文章总结分享,ddd实战代码分享, 领域驱动设计java实战源码大全,我看过的ddd java源码

1. 前段时间研究ddd, 收藏了很多相关知识,分享出来,希望能够帮助更多的小伙伴了解ddd, 什么是领域驱动设计,并分享在github发现的开源ddd代码 2. ddd 必须强烈点赞阿里两位大佬,一个为殷浩, 一个为cola作者 2.1.1 殷浩…

什么是MySql的主从复制(主从同步)?

主页还有其他面试题总结,有需要的可以去看一下,喜欢的就留个三连再走吧~ 1.什么是MySql的主从复制原理? 主从复制的核心就是二进制binlog(DDL(数据定义语言)语句和DML(数据操纵语言&#xff09…

蓝桥云课python代码

第一章语言基础 第一节编程基础 1 python开发环境 第一个Python程序 # 打印"Hello World" print("Hello World")# 打印2的100次方 print(2 ** 100)# 打印112 print("11",1 1)""" Hello World 126765060022822940149670320537…

c#丰田PLC ToyoPuc TCP协议快速读写 to c# Toyota PLC ToyoPuc读写

源代码下载 <------下载地址 历史背景与发展 TOYOPUC协议源于丰田工机&#xff08;TOYODA&#xff09;的自动化技术积累。丰田工机成立于1941年&#xff0c;最初是丰田汽车的机床部门&#xff0c;后独立为专注于工业机械与控制系统的公司。2006年与光洋精工&#xff08;Ko…

深入解析-无状态服务-StatefulSet (一)

一、有状态服务 VS 无状态服务 1.无状态服务介绍 1.数据方面&#xff1a;无状态服务不会在本地存储持久化数据.多个实例可以共享相同的持久化数据 2.结果方面&#xff1a;多个服务实例对于同一个用户请求的响应结果是完全一致的 3.关系方面&#xff1a;这种多服务实例之间是…

hackmyvm-buster

题目地址 信息收集 主机发现 ┌──(root㉿kali)-[/home/kali] └─# arp-scan -I eth1 192.168.56.0/24 Interface: eth1, type: EN10MB, MAC: 00:0c:29:34:da:f5, IPv4: 192.168.56.103 WARNING: Cannot open MAC/Vendor file ieee-oui.txt: Permission denied WARNING: C…

【原创】Windows11安装WSL“无法解析服务器的名称或地址”问题解决方法

原因分析 出现这个问题一开始以为WSL设置了某个服务器&#xff0c;但是通过运行 nslookup www.microsoft.com 出现下面的提示 PS C:\Windows\system32> nslookup www.microsoft.com 服务器: UnKnown Address: 2408:8000:XXXX:2b00:8:8:8:8非权威应答: 名称: e13678…

网页制作08-html,css,javascript初认识のhtml使用框架结构,请先建立站点!

框架一般由框架集和框架组成。 框架集就像一个大的容器&#xff0c;包括所有的框架&#xff0c;是框架的集合。 框架是框架集中一个独立的区域用于显示一个独立的网页文档。 框架集是文件html&#xff0c;它定义一组框架的布局和属性&#xff0c;包括框架的数目&#xff0c;框架…

【Vscode 使用】集合1

一、使用make工具管理工程 windows下&#xff0c;下载mingw64&#xff0c;配置好mingw64\bin 为 Win10系统全局变量后。 在mingw64/bin目录下找到mingw32-make.exe工具。复制一份改名为&#xff1a;make.exe&#xff0c;没错&#xff0c;就是那么简单&#xff0c;mingw64自带m…

文章精读篇——用于遥感小样本语义分割的可学习Prompt

题目&#xff1a;Learnable Prompt for Few-Shot Semantic Segmentation in Remote Sensing Domain 会议&#xff1a;CVPR 2024 Workshop 论文&#xff1a;10.48550/arXiv.2404.10307 相关竞赛&#xff1a;https://codalab.lisn.upsaclay.fr/competitions/17568 年份&#…

解决 kubeasz 安装k8s集群跨节点pod 无法使用cluster ip通讯问题

问题描述 使用kubeasz搭建k8s集群后使用的配置文件 # etcd cluster should have odd member(s) (1,3,5,...) [etcd] 192.168.xx.22# master node(s) [kube_master] 192.168.xx.22# work node(s) [kube_node] 192.168.xx.9 192.168.xx.22# [optional] harbor server, a privat…

Docker 搭建 Nginx 服务器

系列文章目录 Docker 搭建 Nginx 服务器 系列文章目录前言一、准备工作二、设置 Nginx 容器的目录结构三、启动一个临时的 Nginx 容器来复制配置文件四、复制 Nginx 配置文件到本地目录五、删除临时 Nginx 容器六、创建并运行 Nginx 容器&#xff0c;挂载本地目录七、修改 ngin…