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

写屏障和读屏障的区别是什么?


写屏障(Write Barrier)与读屏障(Read Barrier)的区别

在计算机科学中,写屏障读屏障是两种关键的内存同步机制,主要用于解决并发编程中的可见性、有序性问题,或在垃圾回收(GC)中维护内存一致性。它们的核心区别在于触发的操作类型、应用场景及实现目标。以下是详细对比:


1. 定义与核心作用

类型定义核心作用
写屏障写操作前后插入的同步指令或逻辑,确保写操作的顺序性和可见性。防止写操作重排序,确保其他线程能及时看到修改后的值。
读屏障读操作前后插入的同步指令或逻辑,确保读操作的顺序性和数据有效性。防止读操作重排序,确保读取的是最新值或符合预期的状态。

2. 应用场景

(1) 并发编程中的内存屏障
  • 写屏障

    • 场景:在多线程中,当一个线程修改共享变量后,需确保该修改对其他线程可见。
    • 实现
      在写操作后插入StoreStoreStoreLoad屏障,强制将写操作结果刷到主内存。
      示例volatile变量的写操作会自动插入写屏障。
      volatile int x = 1;
      // 写操作后插入StoreLoad屏障,确保写完成且后续读能看到新值
      
  • 读屏障

    • 场景:当一个线程读取共享变量时,需确保读取的是最新值,而非本地缓存的旧值。
    • 实现
      在读操作前插入LoadLoadLoadStore屏障,强制从主内存重新加载数据。
      示例volatile变量的读操作会自动插入读屏障。
      int y = x; // 读volatile变量x,插入LoadLoad屏障确保读取最新值
      
(2) 垃圾回收(GC)中的屏障
  • 写屏障(GC Write Barrier)

    • 场景:在并发标记或移动式GC(如G1、ZGC)中,跟踪对象引用的修改,防止漏标或误标。
    • 实现
      当程序修改对象A的引用指向对象B时,写屏障记录此次修改(如将B加入标记队列)。
      示例:在CMS的并发标记阶段,写屏障用于记录跨代引用。
  • 读屏障(GC Read Barrier)

    • 场景:在增量式GC或并发压缩(如Shenandoah)中,确保读取的引用是有效的。
    • 实现
      当程序读取对象引用时,读屏障检查该引用是否已被移动或无效,必要时触发修复逻辑。
      示例:ZGC使用读屏障实现染色指针,检查引用是否指向有效地址。

3. 底层实现对比

维度写屏障读屏障
触发时机写操作(如赋值、字段更新)后触发。读操作(如加载变量、访问字段)前触发。
硬件指令对应StoreStoreStoreLoad屏障(如x86的mfence)。对应LoadLoadLoadStore屏障(如x86的lfence)。
性能开销较高(需刷写缓存到内存)。较低(仅需刷新本地缓存或加载最新值)。
典型应用- volatile
- 锁释放
- GC中的引用更新跟踪
- volatile
- 锁获取
- GC中的引用有效性检查

4. 实际案例

(1) Java中的volatile变量
  • 写屏障

    volatile int sharedVar = 10;
    // 写操作后插入StoreStore + StoreLoad屏障,确保:
    // 1. 当前线程的写操作对其他线程可见。
    // 2. 禁止与后续操作重排序。
    
  • 读屏障

    int value = sharedVar; // 读操作前插入LoadLoad + LoadStore屏障,确保:
    // 1. 从主内存加载最新值。
    // 2. 禁止与之前操作重排序。
    
(2) 垃圾回收器中的屏障
  • G1 GC的写屏障

    • 当对象A的字段从指向B改为指向C时,写屏障将旧引用B和新引用C加入SATB(Snapshot-At-The-Beginning)队列,供并发标记使用。
  • ZGC的读屏障

    • 读取对象引用时,检查指针元数据(颜色标记),若对象已被移动,则通过读屏障转发到新地址。

5. 性能与权衡

类型优势劣势
写屏障确保数据修改的及时可见性,避免其他线程读取脏数据。频繁写操作时性能损耗较大(如大量volatile写)。
读屏障按需加载最新数据,减少不必要的内存同步开销。读操作可能延迟(需等待屏障逻辑完成)。

尾声

  • 写屏障:关注写操作的有序性与可见性,用于同步数据修改、GC引用跟踪等场景。
  • 读屏障:关注读操作的有序性与数据有效性,用于同步数据加载、GC引用检查等场景。
  • 核心区别
    • 写屏障解决“如何让其他线程看到我的修改”问题。
    • 读屏障解决“如何确保我读到的是最新有效数据”问题。

在这里插入图片描述

http://www.xdnf.cn/news/5489.html

相关文章:

  • 思维链是仅仅通过提示词实现的吗
  • Java对象的内存分布(二)
  • Python训练营打卡——DAY22(2025.5.11)
  • UGMathBench动态基准测试数据集发布 可评估语言模型数学推理能力
  • Maven 中的 pom.xml 文件
  • Mind Over Machines 公司:技术咨询与创新的卓越实践
  • redis存储结构
  • UOJ 164【清华集训2015】V Solution
  • 【C语言】程序的预处理,#define详解
  • 用于文件上传的MultipartFile接口
  • Go语言实现优雅关机和重启的示例
  • 自然语言处理 (NLP) 入门:NLTK 与 SpaCy 的初体验
  • 『 测试 』测试基础
  • nanodet配置文件分析
  • 快速理解动态代理
  • 锚定基础与拥抱融合:C 语言在编程教育与技术社区的破圈之路
  • 长短期记忆网络(LSTM)深度解析:从理论到实践的全方位指南
  • 初探函数使用
  • L1 第9次课 数组进阶
  • 大数据从专家到小白
  • MCP 通信机制:stdio vs SSE
  • 项目过程中使用vant组件使用踩坑记录
  • 【Bootstrap V4系列】学习入门教程之 组件-媒体对象(Media object)
  • Nginx的增强与可视化!OpenResty Manager - 现代化UI+高性能反向代理+安全防护
  • 无人甘蔗小车履带式底盘行走系统的研究
  • 语音合成之十三 中文文本归一化在现代语音合成系统中的应用与实践
  • 【Java学习笔记】instanceof操作符
  • 隐式/显式类型转换?编程语言的类型转换?其它类型转换成数值类型?其它类型转换成字符串?类型转换?
  • 【和春笋一起学C++】数组名作为函数参数实例
  • STM32f103 标准库 零基础学习之按键点灯(不涉及中断)