欧美一区二区三区老妇人-欧美做爰猛烈大尺度电-99久久夜色精品国产亚洲a-亚洲福利视频一区二区

在Java中使用volatile時(shí)需要注意哪些事項(xiàng)-創(chuàng)新互聯(lián)

這篇文章給大家介紹在Java中使用volatile時(shí)需要注意哪些事項(xiàng),內(nèi)容非常詳細(xì),感興趣的小伙伴們可以參考借鑒,希望對大家能有所幫助。

創(chuàng)新互聯(lián)是一家集網(wǎng)站建設(shè),喀什企業(yè)網(wǎng)站建設(shè),喀什品牌網(wǎng)站建設(shè),網(wǎng)站定制,喀什網(wǎng)站建設(shè)報(bào)價(jià),網(wǎng)絡(luò)營銷,網(wǎng)絡(luò)優(yōu)化,喀什網(wǎng)站推廣為一體的創(chuàng)新建站企業(yè),幫助傳統(tǒng)企業(yè)提升企業(yè)形象加強(qiáng)企業(yè)競爭力。可充分滿足這一群體相比中小企業(yè)更為豐富、高端、多元的互聯(lián)網(wǎng)需求。同時(shí)我們時(shí)刻保持專業(yè)、時(shí)尚、前沿,時(shí)刻以成就客戶成長自我,堅(jiān)持不斷學(xué)習(xí)、思考、沉淀、凈化自己,讓我們?yōu)楦嗟钠髽I(yè)打造出實(shí)用型網(wǎng)站。

并發(fā)的三個(gè)特性

首先說我們?nèi)绻褂?volatile 了,那肯定是在多線程并發(fā)的環(huán)境下。我們常說的并發(fā)場景下有三個(gè)重要特性:原子性、可見性、有序性。只有在滿足了這三個(gè)特性,才能保證并發(fā)程序正確執(zhí)行,否則就會出現(xiàn)各種各樣的問題。

原子性,上篇文章說到的 CAS 和 Atomic* 類,可以保證簡單操作的原子性,對于一些負(fù)責(zé)的操作,可以使用synchronized 或各種鎖來實(shí)現(xiàn)。

可見性,指當(dāng)多個(gè)線程訪問同一個(gè)變量時(shí),一個(gè)線程修改了這個(gè)變量的值,其他線程能夠立即看得到修改的值。

有序性,程序執(zhí)行的順序按照代碼的先后順序執(zhí)行,禁止進(jìn)行指令重排序。看似理所當(dāng)然的事情,其實(shí)并不是這樣,指令重排序是JVM為了優(yōu)化指令,提高程序運(yùn)行效率,在不影響單線程程序執(zhí)行結(jié)果的前提下,盡可能地提高并行度。但是在多線程環(huán)境下,有些代碼的順序改變,有可能引發(fā)邏輯上的不正確。

而 volatile 做實(shí)現(xiàn)了兩個(gè)特性,可見性和有序性。所以說在多線程環(huán)境中,需要保證這兩個(gè)特性的功能,可以使用 volatile 關(guān)鍵字。

volatile 是如何保證可見性的

說到可見性,就要了解一下計(jì)算機(jī)的處理器和主存了。因?yàn)槎嗑€程,不管有多少個(gè)線程,最后還是要在計(jì)算機(jī)處理器中進(jìn)行的,現(xiàn)在的計(jì)算機(jī)基本都是多核的,甚至有的機(jī)器是多處理器的。我們看一下多處理器的結(jié)構(gòu)圖:

在Java中使用volatile時(shí)需要注意哪些事項(xiàng)

這是兩個(gè)處理器,四核的 CPU。一個(gè)處理器對應(yīng)一個(gè)物理插槽,多處理器間通過QPI總線相連。一個(gè)處理器包含多個(gè)核,一個(gè)處理器間的多核共享L3 Cache。一個(gè)核包含寄存器、L1 Cache、L2 Cache。

在程序執(zhí)行的過程中,一定要涉及到數(shù)據(jù)的讀和寫。而我們都知道,雖然內(nèi)存的訪問速度已經(jīng)很快了,但是比起CPU執(zhí)行指令的速度來,還是差的很遠(yuǎn)的,因此,在內(nèi)核中,增加了L1、L2、L3 三級緩存,這樣一來,當(dāng)程序運(yùn)行的時(shí)候,先將所需要的數(shù)據(jù)從主存復(fù)制一份到所在核的緩存中,運(yùn)算完成后,再寫入主存中。下圖是 CPU 訪問數(shù)據(jù)的示意圖,由寄存器到高速緩存再到主存甚至硬盤的速度是越來越慢的。

在Java中使用volatile時(shí)需要注意哪些事項(xiàng)

了解了 CPU 結(jié)構(gòu)之后,我們來看一下程序執(zhí)行的具體過程,拿一個(gè)簡單的自增操作舉例。

i=i+1;

執(zhí)行這條語句的時(shí)候,在某個(gè)核上運(yùn)行的某線程將 i 的值拷貝一個(gè)副本到此核所在的緩存中,當(dāng)運(yùn)算執(zhí)行完成后,再回寫到主存中去。如果是多線程環(huán)境下,每一個(gè)線程都會在所運(yùn)行的核上的高速緩存區(qū)有一個(gè)對應(yīng)的工作內(nèi)存,也就是每一個(gè)線程都有自己的私有工作緩存區(qū),用來存放運(yùn)算需要的副本數(shù)據(jù)。那么,我們再來看這個(gè) i+1 的問題,假設(shè) i 的初始值為0,有兩個(gè)線程同時(shí)執(zhí)行這條語句,每個(gè)線程執(zhí)行都需要三個(gè)步驟:

1、從主存讀取 i 值到線程工作內(nèi)存,也就是對應(yīng)的內(nèi)核高速緩存區(qū);

2、計(jì)算 i+1 的值;

3、將結(jié)果值寫回主存中;

建設(shè)兩個(gè)線程各執(zhí)行 10,000 次后,我們預(yù)期的值應(yīng)該是 20,000 才對,可惜很遺憾,i 的值總是小于 20,000 的 。導(dǎo)致這個(gè)問題的其中一個(gè)原因就是緩存一致性問題,對于這個(gè)例子來說,一旦某個(gè)線程的緩存副本做了修改,其他線程的緩存副本應(yīng)該立即失效才對。

而使用了 volatile 關(guān)鍵字后,會有如下效果:

1、每次對變量的修改,都會引起處理器緩存(工作內(nèi)存)寫回到主存;

2、一個(gè)工作內(nèi)存回寫到主存會導(dǎo)致其他線程的處理器緩存(工作內(nèi)存)無效。

因?yàn)?volatile 保證內(nèi)存可見性,其實(shí)是用到了 CPU 保證緩存一致性的 MESI 協(xié)議。MESI 協(xié)議內(nèi)容較多,這里就不做說明,請各位同學(xué)自己去查詢一下吧??傊昧?volatile 關(guān)鍵字,當(dāng)某線程對 volatile 變量的修改會立即回寫到主存中,并且導(dǎo)致其他線程的緩存行失效,強(qiáng)制其他線程再使用變量時(shí),需要從主存中讀取。

那么我們把上面的 i 變量用 volatile 修飾后,再次執(zhí)行,每個(gè)線程執(zhí)行 10,000 次。很遺憾,還是小于 20,000 的。這是為什么呢?

volatile 利用 CPU 的 MESI 協(xié)議確實(shí)保證了可見性。但是,注意了,volatile 并沒有保證操作的原子性,因?yàn)檫@個(gè)自增操作是分三步的,假設(shè)線程 1 從主存中讀取了 i 值,假設(shè)是 10 ,并且此時(shí)發(fā)生了阻塞,但是還沒有對i進(jìn)行修改,此時(shí)線程 2 也從主存中讀取了 i 值,這時(shí)這兩個(gè)線程讀取的 i 值是一樣的,都是 10 ,然后線程 2 對 i 進(jìn)行了加 1 操作,并立即寫回主存中。此時(shí),根據(jù) MESI 協(xié)議,線程 1 的工作內(nèi)存對應(yīng)的緩存行會被置為無效狀態(tài),沒錯(cuò)。但是,請注意,線程 1 早已經(jīng)將 i 值從主存中拷貝過了,現(xiàn)在只要執(zhí)行加 1 操作和寫回主存的操作了。而這兩個(gè)線程都是在 10 的基礎(chǔ)上加 1 ,然后又寫回主存中,所以最后主存的值只是 11 ,而不是預(yù)期的 12 。

所以說,使用 volatile 可以保證內(nèi)存可見性,但無法保證原子性,如果還需要原子性,可以參考,之前的這篇文章。

volatile 是如何保證有序性的

Java 內(nèi)存模型具備一些先天的“有序性”,即不需要通過任何手段就能夠得到保證的有序性,這個(gè)通常也稱為 happens-before 原則。如果兩個(gè)操作的執(zhí)行次序無法從 happens-before 原則推導(dǎo)出來,那么它們就不能保證它們的有序性,虛擬機(jī)可以隨意地對它們進(jìn)行重排序。

如下是 happens-before 的8條原則,摘自 《深入理解Java虛擬機(jī)》。

  • 程序次序規(guī)則:一個(gè)線程內(nèi),按照代碼順序,書寫在前面的操作先行發(fā)生于書寫在后面的操作;

  • 鎖定規(guī)則:一個(gè) unLock 操作先行發(fā)生于后面對同一個(gè)鎖的 lock 操作;

  • volatile 變量規(guī)則:對一個(gè)變量的寫操作先行發(fā)生于后面對這個(gè)變量的讀操作;

  • 傳遞規(guī)則:如果操作A先行發(fā)生于操作B,而操作B又先行發(fā)生于操作C,則可以得出操作A先行發(fā)生于操作C;

  • 線程啟動規(guī)則:Thread對象的start()方法先行發(fā)生于此線程的每個(gè)一個(gè)動作;

  • 線程中斷規(guī)則:對線程interrupt()方法的調(diào)用先行發(fā)生于被中斷線程的代碼檢測到中斷事件的發(fā)生;

  • 線程終結(jié)規(guī)則:線程中所有的操作都先行發(fā)生于線程的終止檢測,我們可以通過Thread.join()方法結(jié)束、Thread.isAlive()的返回值手段檢測到線程已經(jīng)終止執(zhí)行;

  • 對象終結(jié)規(guī)則:一個(gè)對象的初始化完成先行發(fā)生于他的 finalize() 方法的開始;

這里主要說一下 volatile 關(guān)鍵字的規(guī)則,舉一個(gè)著名的單例模式中的雙重檢查的例子:

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

如果 instance 不用 volatile 修飾,可能產(chǎn)生什么結(jié)果呢,假設(shè)有兩個(gè)線程在調(diào)用 getInstance() 方法,線程 1 執(zhí)行步驟 step1 ,發(fā)現(xiàn) instance 為 null ,然后同步鎖住 Singleton 類,接著再次判斷 instance 是否為 null ,發(fā)現(xiàn)仍然是 null,然后執(zhí)行 step 3 ,開始實(shí)例化 Singleton 。而在實(shí)例化的過程中,線程 2 走到 step 1,有可能發(fā)現(xiàn) instance 不為空,但是此時(shí) instance 有可能還沒有完全初始化。

什么意思呢,對象在初始化的時(shí)候分三個(gè)步驟,用下面的偽代碼表示:

memory = allocate(); //1. 分配對象的內(nèi)存空間 
ctorInstance(memory); //2. 初始化對象
instance = memory; //3. 設(shè)置 instance 指向?qū)ο蟮膬?nèi)存空間

因?yàn)椴襟E 2 和步驟 3 需要依賴步驟 1,而步驟 2 和 步驟 3 并沒有依賴關(guān)系,所以這兩條語句有可能會發(fā)生指令重排,也就是或有可能步驟 3 在步驟 2 的之前執(zhí)行。在這種情況下,步驟 3 執(zhí)行了,但是步驟 2 還沒有執(zhí)行,也就是說 instance 實(shí)例還沒有初始化完畢,正好,在此刻,線程 2 判斷 instance 不為 null,所以就直接返回了 instance 實(shí)例,但是,這個(gè)時(shí)候 instance 其實(shí)是一個(gè)不完全的對象,所以,在使用的時(shí)候就會出現(xiàn)問題。

而使用 volatile 關(guān)鍵字,也就是使用了 “對一個(gè) volatile修飾的變量的寫,happens-before于任意后續(xù)對該變量的讀” 這一原則,對應(yīng)到上面的初始化過程,步驟2 和 3 都是對 instance 的寫,所以一定發(fā)生于后面對 instance 的讀,也就是不會出現(xiàn)返回不完全初始化的 instance 這種可能。

JVM 底層是通過一個(gè)叫做“內(nèi)存屏障”的東西來完成。內(nèi)存屏障,也叫做內(nèi)存柵欄,是一組處理器指令,用于實(shí)現(xiàn)對內(nèi)存操作的順序限制。

關(guān)于在Java中使用volatile時(shí)需要注意哪些事項(xiàng)就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,可以學(xué)到更多知識。如果覺得文章不錯(cuò),可以把它分享出去讓更多的人看到。

分享題目:在Java中使用volatile時(shí)需要注意哪些事項(xiàng)-創(chuàng)新互聯(lián)
轉(zhuǎn)載注明:http://chinadenli.net/article44/ceppee.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供移動網(wǎng)站建設(shè)、營銷型網(wǎng)站建設(shè)外貿(mào)網(wǎng)站建設(shè)、靜態(tài)網(wǎng)站外貿(mào)建站、網(wǎng)站建設(shè)

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)

成都定制網(wǎng)站建設(shè)
成人午夜激情在线免费观看| 我的性感妹妹在线观看| 国产精品久久精品毛片| 亚洲欧美日韩国产成人| 亚洲一区二区三区在线免费| 欧美激情一区二区亚洲专区| 午夜福利直播在线视频| 国产一区二区三区不卡| 亚洲人午夜精品射精日韩| 欧美日韩乱一区二区三区| 日韩欧美三级中文字幕| 日韩精品区欧美在线一区| 欧美区一区二在线播放| 日韩国产欧美中文字幕| 欧美野外在线刺激在线观看| 日本办公室三级在线观看| 免费观看日韩一级黄色大片| 久久偷拍视频免费观看| 最近最新中文字幕免费| 色婷婷成人精品综合一区| 熟女少妇一区二区三区蜜桃| 久久精品一区二区少妇| 色丁香一区二区黑人巨大| 日韩精品少妇人妻一区二区| 国产精品第一香蕉视频| 国产精品午夜小视频观看| 男人的天堂的视频东京热| 午夜日韩在线观看视频| 伊人欧美一区二区三区| 日本免费一本一二区三区| 蜜桃传媒在线正在播放| 日韩一区二区三区18| 久久精品偷拍视频观看| 日韩精品一区二区三区四区| 中文字幕佐山爱一区二区免费| 亚洲精品小视频在线观看| 日韩人妻毛片中文字幕| 欧美午夜一级艳片免费看| 少妇人妻中出中文字幕| 草草视频福利在线观看| 久久99一本色道亚洲精品|