volatile關鍵值

来源:https://www.cnblogs.com/kundeg/archive/2018/03/12/8552124.html
-Advertisement-
Play Games

happens before原則 我們編寫的程式都要經過優化後(編譯器和處理器會對我們的程式進行優化以提高運行效率)才會被運行,優化分為很多種,其中有一種優化叫做重排序,重排序需要遵守happens before規則,換句話說只要滿足happens before原則就可以進行重排序。 定義 :在JM ...


happens-before原則

我們編寫的程式都要經過優化後(編譯器和處理器會對我們的程式進行優化以提高運行效率)才會被運行,優化分為很多種,其中有一種優化叫做重排序,重排序需要遵守happens-before規則,換句話說只要滿足happens-before原則就可以進行重排序。

定義:在JMM中,如果一個操作執行的結果需要對另一個操作可見,那麼這兩個操作之間必須存在happens-before關係

註意:定義中所說的前一個操作happens-before後一個操作並不是說前一個操作必須要在後一個操作之前執行,而是指前一個操作的執行結果必須對後一個操作可見,考慮下述情況:

int a = 1; //操作A
int b = 2; //操作B

單線程執行上述代碼塊規定操作A happens-before 操作B,也就是說操作A的結果對操作B是可見的,但是操作B對操作A中a=1的賦值並沒有依賴,即使操作A與操作B重排序了,它們之間的happens-before關係仍然存在,這個例子就說明瞭happens-before並不是對執行順序對約束,同時也是重排序的一種情況。

規則:

  • 程式次序規則:一個線程內,按照代碼順序,書寫在前面的操作先行發生於書寫在後面的操作;
  • 鎖定規則:一個unLock操作先行發生於後面對同一個鎖的lock操作;
  • volatile變數規則:對一個變數的寫操作先行發生於後面對這個變數的讀操作;
  • 傳遞規則:如果操作A先行發生於操作B,而操作B又先行發生於操作C,則可以得出操作A先行發生於操作C;
  • 線程啟動規則:Thread對象的start()方法先行發生於此線程的每個一個動作;
  • 線程中斷規則:對線程interrupt()方法的調用先行發生於被中斷線程的代碼檢測到中斷事件的發生;
  • 線程終結規則:線程中所有的操作都先行發生於線程的終止檢測,我們可以通過Thread.join()方法結束、Thread.isAlive()的返回值手段檢測到線程已經終止執行;
  • 對象終結規則:一個對象的初始化完成先行發生於他的finalize()方法的開始;

volatile關鍵字

可見性

volatile修飾的變數的一個特點是可見性:保證被volatile修飾的共用變數對所有線程可見,也就是當一個線程修改了一個被volatile修飾變數的值,其他線程可以立即得知新值,舉例:

volatile boolean shutdownRequested;
public void shutdown(){
    shutdownRequested = true;
}

public void doWork(){
    while(!shutdownRequested){
        //do stuff
    }
}

錯誤用法:

public class VolatileVisibility {
    public static volatile int i =0;

    public static void increase(){
        i++;
    }
}
/**
volatile關鍵值不保證有序性,i++包括讀取一個值,然後寫回一個新值,新值比原來值加了1,這相當於兩個步驟,如果第二個線程在第一個線程讀取舊值和寫回新值期間讀取i的值,併進行加一操作,會發生更新重覆,存線上程安全問題
**/

有序性

volatile修飾的變數的另一個特是有序性點:禁止指令重排優化,從而避免多線程環境下程式出現亂序執行的現象

//雙重校驗鎖
public class Singleton {
private static volatile Singleton instance = null;
private Singleton() { }
public static Singleton getInstance() {
    if(instance == null) {
        synchronized(Singleton.class) {
            if(instance == null) {
                instance = new Singleton();
            }
        }
      }
    return instance;
  }
}

疑問:上述代碼Singleton變數為什麼要用volatile修飾?
解答:
instance = new Singleton()可以分為下述步驟完成:

memory = allocate();   //1:分配對象的記憶體空間  
instance(memory);      //2:初始化對象  
instance = memory;     //3:設置instance指向剛分配的記憶體地址  

由於2,3步驟沒有數據依賴關係,因此2,3可以重排序並沒有違背單線程的happens-before規則,重排後如下:

memory = allocate(); //1.分配對象記憶體空間
instance = memory;   //3.設置instance指向剛分配的記憶體地址,此時instance!=null,但是對象還沒有初始化完成!
instance(memory);    //2.初始化對象

根據volatile變數的可見性,在執行完3後,instance不為空,但是尚未實例化,但是此時如果有線程過來請求實例,就可能返回尚未實例化對象。

記憶體屏障

緩存一致性

  • 嗅探機制(snooping):所有記憶體傳輸都發生在一條共用的匯流排上,而所有的處理器都能看到這條匯流排:緩存本身是獨立的,但是記憶體是共用資源,嗅探(snooping)協議的思想是,緩存不僅僅在做記憶體傳輸的時候才和匯流排打交道,而是不停地在嗅探匯流排上發生的數據交換,跟蹤其他緩存在做什麼。所以當一個緩存代表它所屬的處理器去讀寫記憶體時,其他處理器都會得到通知,它們以此來使自己的緩存保持同步。只要某個處理器一寫記憶體,其他處理器馬上就知道這塊記憶體在它們自己的緩存中對應的段已經失效。

  • 匯流排鎖機制(lock):在指令前面加上lock,那麼會鎖住匯流排和相應的緩存,其他指令會被阻塞,當lock後的指令執行完畢會將結果刷新到記憶體中去,根據嗅探機制,其他cpu中的緩存會失效,重新從記憶體中讀取,也就解決了緩存一致性問題

  • 緩存一致性協議(MESI):cpu緩存有四個標記位:
    M: Modify,修改緩存,當前CPU的緩存已經被修改了,即與記憶體中數據已經不一致了
    E: Exclusive,獨占緩存,當前CPU的緩存和記憶體中數據保持一致,而且其他處理器並沒有可使用的緩存數據
    S: Share,共用緩存,和記憶體保持一致的一份拷貝,多組緩存可以同時擁有針對同一記憶體地址的共用緩存段
    I: Invalid,失效緩存,這個說明CPU中的緩存已經不能使用了
    CPU的讀取遵循下麵幾點:
    如果緩存狀態是I,那麼就從記憶體中讀取,否則就從緩存中直接讀取。
    如果緩存處於M或E的CPU讀取到其他CPU有讀操作,就把自己的緩存寫入到記憶體中,並將自己的狀態設置為S。
    只有緩存狀態是M或E的時候,CPU才可以修改緩存中的數據,將其他cpu緩存設置無效,修改後,緩存狀態變為M

記憶體屏障

  • 硬體層的記憶體屏障分為兩種:Load Barrier 和 Store Barrier即讀屏障和寫屏障。
  • 記憶體屏障有兩個作用:

阻止屏障兩側的指令重排序;
強制把寫緩衝區/高速緩存中的臟數據等寫回主記憶體,讓緩存中相應的數據失效。

  • 對於Load Barrier來說,在指令前插入LoadBarrier,可以讓高速緩存中的數據失效,強制從新從主記憶體載入數據;對應的在讀volatile變數前加上Lfence
  • 對於Store Barrier來說,在指令後插入StoreBarrier,能讓寫入緩存中的最新數據更新寫入主記憶體,讓其他線程可見,對應的在寫volatile變數後加上Sfence

參考資料

http://blog.csdn.net/u010031673/article/details/48153797
https://kb.cnblogs.com/page/504824/
https://www.cnblogs.com/dolphin0520/p/3920373.html
https://www.jianshu.com/p/195ae7c77afe
http://blog.csdn.net/iter_zc/article/details/42006811
https://www.jianshu.com/p/2ab5e3d7e510


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • Description 已知va和vb分別為非遞減有序線性表,將va和vb進行合併為新的線性表vc,並保持vc仍然非遞減有序。 本題中,線性表元素為整數。線性表的最大長度為1000。 Input 輸入數據有多組,第一行為測試數據的組數t,接下來為2t行,每一組測試數據有兩行: 第一行的第一個數為va ...
  • 編寫第一個Java程式 完成工作:1.在文本編輯器中輸入一個Java程式。 2.使用括弧組織程式。 3.保存、編譯和運行程式。 1 package com.Jsample;//將程式的包名稱命名為com.Jsample 2 3 public class Helloworld {//將程式(類)命名為 ...
  • 列表函數 追加和擴展 list.append() 在列表末尾追加新的對象 extend()在列表末尾一次性追加另一個序列中的多個值(用新列表擴展原來的列表) 其他函數 count() 統計某個元素在列表中出現的次數 index() 從列表中找出某個值第一個匹配項的索引位置 insert() 將對象插 ...
  • 1 Celery簡介 Celery是非同步任務隊列,可以獨立於主進程運行,在主進程退出後,也不影響隊列中的任務執行。 任務執行異常退出,重新啟動後,會繼續執行隊列中的其他任務,同時可以緩存停止期間接收的工作任務,這個功能依賴於消息隊列(MQ、Redis)。 1.1 Celery原理 Celery的架構 ...
  • 1、方法的定義格式及解析 (1)方法概述:方法就是完成特定功能的代碼塊。 (2)定義格式: 修飾符 返回值類型 方法名(參數1,參數2,參數3...){ 函數體; return 返回值; } (3)修飾符:公共類public、私有類private、抽象類abstract、最終類final。 (4)返 ...
  • 測試環境:Keil 5.20.0.0 STM32F103RBT6 固件庫版本:STM32F10x_StdPeriph_Lib_V3.5.0(2011) 本文使用TIM1的通道1,通道2,產生兩路1khz,死區時間1us的互補PWM波。 所使用的IO口:由下圖知,我們使用引腳為PA9,PA10,互補輸 ...
  • 一.java常用數據類型 int 只有 true或false沒有0或非0 二.數據類型轉換 1.自動轉換:byte ->short int->char->int->long int ->float->double 轉換條件:由低類型向高類型(即箭頭所指的轉換方向)變數類型不會改變,但計算值會變為高類 ...
  • 題目描述 過長……不想發圖也不想發文字,所以就發鏈接吧…… [沒有人的算術][1] 題解 $orz$神題一枚 我們考慮如果插入的數不是數對,而是普通的數,這就是一道傻題了——直接線段樹一頓亂上就可以了。 於是我們現在只需要解決一個問題——維護這些數的大小關係。 由於這些數具有有序性,我們可以將這些數 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...