欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

一文揭秘Java內(nèi)存模型的隱匿陷阱與解決方案

 更新時(shí)間:2025年06月23日 08:42:29   作者:悟能不能悟  
這篇文章主要為大家詳細(xì)介紹了Java中內(nèi)存模型的隱匿陷阱與解決方案,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下

問題背景

資深Java面試題:??

“假設(shè)存在以下基于volatile的并發(fā)代碼:

public class VolatileExample {
    private volatile boolean flag = false;
    private int counter = 0;
 
    public void writer() {
        counter = 42;      // 非volatile寫
        flag = true;       // volatile寫
    }
 
    public void reader() {
        if (flag) {         // volatile讀
            System.out.println(counter); // 輸出什么?
        }
    }
}

問:當(dāng)兩個(gè)線程分別調(diào)用writer()和reader()時(shí),reader()方法是否可能輸出0?為什么?如何修正?”

技術(shù)解析與博客正文

1. 問題答案:是的,可能輸出0!

看似volatile的flag保證可見性,但JMM(Java內(nèi)存模型)對(duì)非volatile變量的語義約束是破局關(guān)鍵:

?volatile寫?(flag=true)僅保證其之前的普通寫(counter=42)不會(huì)被重排序到其后?(StoreStore屏障)??

?volatile讀?(if(flag))僅保證其之后的普通讀(counter)不會(huì)被重排序到其前?(LoadLoad屏障)??

?但普通寫(counter=42)與普通讀(counter)之間無任何同步保證!??

若counter=42因CPU緩存未刷新、編譯器優(yōu)化等原因延遲對(duì)reader()可見,則輸出0成為可能。

2. 深度探因:CPU緩存架構(gòu)與內(nèi)存屏障

?CPU緩存不一致性?:當(dāng)writer()線程在Core1執(zhí)行,counter=42可能僅寫入Core1的L1緩存,尚未同步至主存。

?編譯器和CPU的重排序?:為提高性能,指令可能被重新排序(只要符合as-if-serial語義)。

?volatile的語義局限性?:僅對(duì)自身和關(guān)聯(lián)操作提供有限屏障,而非保證全部變量可見性。

3. 解決方案對(duì)比

方案1: 所有共享變量加volatile(不推薦)

private volatile int counter = 0;

?缺點(diǎn)?:破壞封裝性,且大量volatile寫降低性能(強(qiáng)制緩存一致性協(xié)議全程運(yùn)行)。

方案2: 鎖同步(synchronized)

public synchronized void writer() { ... }
public synchronized void reader() { ... }

?缺點(diǎn)?:重量級(jí)操作,線程阻塞帶來上下文切換開銷。

方案3: ?JDK 9+ VarHandle:精細(xì)化內(nèi)存屏障控制?

private static final VarHandle COUNTER_HANDLE;
static {
    try {
        COUNTER_HANDLE = MethodHandles
            .lookup()
            .findVarHandle(VolatileExample.class, "counter", int.class);
    } catch (Exception e) { throw new Error(e); }
}
 
public void reader() {
    if (flag) {
        // 顯式插入讀屏障
        COUNTER_HANDLE.loadLoadFence(); 
        System.out.println(counter);
    }
}

?優(yōu)勢(shì)?:

細(xì)粒度控制(僅需在關(guān)鍵位置插入屏障)

避免鎖開銷

兼容Java 9+新特性(如Opaque、Release-Acquire等內(nèi)存模式)

4. 終極方案:java.util.concurrent工具類

private final AtomicInteger counter = new AtomicInteger(0);
 
public void writer() {
    counter.set(42);      // 內(nèi)部包含volatile語義
    flag = true;
}
 
public void reader() {
    if (flag) {
        System.out.println(counter.get()); // 安全!
    }
}

?原理?:

AtomicInteger利用volatile + CAS操作,既保證可見性又避免鎖競(jìng)爭(zhēng)。

5. 驗(yàn)證工具:JcStress框架

@JCStressTest
@Outcome(id = "0", expect = Expect.ACCEPTABLE_INTERESTING, desc = "!!! 可見性失效 !!!")
@Outcome(id = "42", expect = Expect.ACCEPTABLE, desc = "正??梢?)
public class VolatileTest {
    private boolean flag = false;
    private int counter = 0;
 
    @Actor
    public void writer() {
        counter = 42;
        flag = true;
    }
 
    @Actor
    public void reader(IntResult1 r) {
        if (flag) r.r1 = counter;
    }
}

?結(jié)果輸出?:

*** INTERESTING tests
  0 matching test results (僅部分運(yùn)行環(huán)境出現(xiàn))

結(jié)語

“volatile是并發(fā)編程的‘有限承諾’,而非‘萬能 鑰匙’。

理解JMM的 ?Happens-Before原則與內(nèi)存屏障的物理本質(zhì),才能在分布式緩存、NUMA架構(gòu)等復(fù)雜場(chǎng)景中游刃有余。

推薦策略:

  • 優(yōu)先使用java.util.concurrent原子類
  • 高并發(fā)場(chǎng)景考慮VarHandle精確控制
  • 復(fù)雜狀態(tài)機(jī)使用StampedLock等新型鎖

忘掉‘我以為’,用JcStress實(shí)測(cè)并發(fā)行為——這是資深工程師的理性修養(yǎng)。”

到此這篇關(guān)于一文揭秘Java內(nèi)存模型的隱匿陷阱與解決方案的文章就介紹到這了,更多相關(guān)Java內(nèi)存模型內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 詳解SpringBoot中@SessionAttributes的使用

    詳解SpringBoot中@SessionAttributes的使用

    這篇文章主要通過示例為大家詳細(xì)介紹了SpringBoot中@SessionAttributes的使用,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解一下
    2022-07-07
  • 一天時(shí)間用Java寫了個(gè)飛機(jī)大戰(zhàn)游戲,朋友直呼高手

    一天時(shí)間用Java寫了個(gè)飛機(jī)大戰(zhàn)游戲,朋友直呼高手

    前兩天我發(fā)現(xiàn)論壇有兩篇飛機(jī)大戰(zhàn)的文章異?;鸨?但都是python寫的,竟然不是我大Java,說實(shí)話作為老java選手,我心里是有那么一些失落的,今天特地整理了這篇文章,需要的朋友可以參考下
    2021-05-05
  • Java設(shè)計(jì)模式之訪問者模式

    Java設(shè)計(jì)模式之訪問者模式

    這篇文章介紹了Java設(shè)計(jì)模式之訪問者模式,文中通過示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2022-10-10
  • Java設(shè)計(jì)模式之初識(shí)行為型模式

    Java設(shè)計(jì)模式之初識(shí)行為型模式

    今天帶大家學(xué)習(xí)Java設(shè)計(jì)模式的相關(guān)知識(shí)點(diǎn),文中對(duì)Java行為型模式做了非常詳細(xì)的介紹及代碼示例,對(duì)正在學(xué)習(xí)java的小伙伴們很有幫助,需要的朋友可以參考下
    2021-06-06
  • SpringBoot+MDC實(shí)現(xiàn)鏈路調(diào)用日志的方法

    SpringBoot+MDC實(shí)現(xiàn)鏈路調(diào)用日志的方法

    MDC是 log4j 、logback及l(fā)og4j2 提供的一種方便在多線程條件下記錄日志的功能,這篇文章主要介紹了SpringBoot+MDC實(shí)現(xiàn)鏈路調(diào)用日志,需要的朋友可以參考下
    2022-12-12
  • springboot集成RestTemplate及常見的用法說明

    springboot集成RestTemplate及常見的用法說明

    這篇文章主要介紹了springboot集成RestTemplate及常見的用法說明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-10-10
  • 線程池FutureTask異步執(zhí)行多任務(wù)實(shí)現(xiàn)詳解

    線程池FutureTask異步執(zhí)行多任務(wù)實(shí)現(xiàn)詳解

    這篇文章主要為大家介紹了線程池FutureTask異步執(zhí)行多任務(wù)實(shí)現(xiàn)詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-11-11
  • Java讓泛型實(shí)例化的方法

    Java讓泛型實(shí)例化的方法

    這篇文章主要介紹了Java讓泛型實(shí)例化的方法,文中示例代碼非常詳細(xì),幫助大家更好的理解和學(xué)習(xí),感興趣的朋友可以了解下
    2020-07-07
  • JdbcTemplate方法介紹與增刪改查操作實(shí)現(xiàn)

    JdbcTemplate方法介紹與增刪改查操作實(shí)現(xiàn)

    這篇文章主要給大家介紹了關(guān)于JdbcTemplate方法與增刪改查操作實(shí)現(xiàn)的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者使用JdbcTemplate具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-11-11
  • springboot中請(qǐng)求路徑配置在配置文件中詳解

    springboot中請(qǐng)求路徑配置在配置文件中詳解

    這篇文章主要介紹了springboot中請(qǐng)求路徑配置在配置文件中,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-01-01

最新評(píng)論