Java并發(fā)之volatile關(guān)鍵字-創(chuàng)新互聯(lián)

引言
說到多線程,我覺得我們最重要的是要理解一個臨界區(qū)概念。

成都創(chuàng)新互聯(lián)長期為成百上千客戶提供的網(wǎng)站建設(shè)服務(wù),團(tuán)隊從業(yè)經(jīng)驗10年,關(guān)注不同地域、不同群體,并針對不同對象提供差異化的產(chǎn)品和服務(wù);打造開放共贏平臺,與合作伙伴共同營造健康的互聯(lián)網(wǎng)生態(tài)環(huán)境。為桑日企業(yè)提供專業(yè)的成都網(wǎng)站建設(shè)、成都做網(wǎng)站,桑日網(wǎng)站改版等技術(shù)服務(wù)。擁有十年豐富建站經(jīng)驗和眾多成功案例,為您定制開發(fā)。

Java并發(fā)之volatile關(guān)鍵字

舉個例子,一個班上1個女孩子(臨界區(qū)),49個男孩子(線程),男孩子的目標(biāo)就是這一個女孩子,就是會有競爭關(guān)系(線程安全問題)。推廣到實際場景,例如對一個數(shù)相加或者相減等等情形,因為操作對象就只有一個,在多線程環(huán)境下,就會產(chǎn)生線程安全問題。理解臨界區(qū)概念,我們對多線程問題可以有一個好意識。

Jav內(nèi)存模型(JMM)

談到多線程就應(yīng)該了解一下Java內(nèi)存模型(JMM)的抽象示意圖.下圖:
Java并發(fā)之volatile關(guān)鍵字

線程A和線程B執(zhí)行的是時候,會去讀取共享變量(臨界區(qū)),然后各自拷貝一份回到自己的本地內(nèi)存,執(zhí)行后續(xù)操作。
JMM模型是一種規(guī)范,就像Java的接口一樣。JMM會涉及到三個問題:原子性,可見性,有序性。
所謂原子性。就是說一個線程的執(zhí)行會不會被其他線程影響的。他是不可中斷的。舉個例子:

int i=1
這個語句在Jmm中就是原子性的。無論是一個線程執(zhí)行還是多個線程執(zhí)行這個語句,讀出來的i就是等于1。那什么是非原子性呢,按道理如果Java的代碼都是原子性,應(yīng)該就不會有線程問題了啊。其實JMM這是規(guī)定某些語句是原子性罷了。舉個非原子性例子:

i ++;
這個操作就不是原子性的了。因為他就是包含了三個操作:第一讀取i的值,第二將i加上1,第三將結(jié)果賦值回來給i,更新i的值。
所謂可見性??梢娦员硎救绻粋€值在線程A修改了,線程B就會馬上知道這個結(jié)果。
所謂有序性。所謂有序性值的是語意的有序性。就是說代碼順序可能會發(fā)生變化。因為有一個指令重排機制。所謂指令重排,他會改變代碼執(zhí)行順序,為了讓cpu執(zhí)行效率更高。為了防止重排序出錯,JMM有個happen-before規(guī)則,這個規(guī)則限制了那些語句執(zhí)行在前,那些語句執(zhí)行在后。
Happen-before:
程序順序原則:一個線程內(nèi)保證語義的串行性
volatile原則:volatile變量的寫發(fā)生在讀之前
鎖規(guī)則:先加鎖再解鎖
傳遞性:a先于b,b先于c,則a必定先于c
線程的start方法先于他的每一個操作
線程所有的操作先于線程的終結(jié)
對象的構(gòu)造函數(shù)執(zhí)行、結(jié)束先于finalize()方法。

volatile
進(jìn)入正題,volatile可以保證變量(臨界區(qū))的可見性以及有序性,但是不能保證原子性。舉個例子:

public class VolatileTest implements Runnable{
    private static VolatileTest volatileTest = new VolatileTest();
    private  static volatile int i= 0;
    public static void main(String[] args) throws InterruptedException {
        for (int j = 0; j < 20; j++) {
            Thread a = new Thread(new VolatileTest());
            Thread b = new Thread(new VolatileTest());
            a.start();b.start();
            a.join();b.join();
            System.out.print(i+"&&");
        }

    }

    @Override
    public void run() {
        for (int j = 0; j < 1000; j++) {
            i++;
        }
    }

}

// 輸出結(jié)果
// 2000&&4000&&5852&&7852&&9852&&11852&&13655&&15655&&17655&&19655&&21306     
//&&22566&&24566&&26189&&28189&&30189&&32189&&34189&&36189&&38089&&

有結(jié)果看到有問題,雖然i已經(jīng)添加了volatile關(guān)鍵字,說明volatile關(guān)鍵字不能保證i++的原子性。

那什么場景適合使用volatile關(guān)鍵字

1,輕量級的“讀-寫鎖”策略

private volatile int value;
public int getValue(){ return value;}
public synchronized void doubleValue(){ value = value*value; }

2.單例模式(雙檢查鎖機制

private volatile static Singleton instace;   
public static Singleton getInstance(){  // 沒有使用同步方法,而是同步方法塊
    //第一次null檢查 ,利用volatile的線程間可見性,不需要加鎖,性能提高    
    if(instance == null){            
        synchronized(Singleton.class) {    //鎖住類對象,阻塞其他線程
            //第二次null檢查,以保證不會創(chuàng)建重復(fù)的實例       
            if(instance == null){       
                instance = new Singleton(); // 禁止重排序
            }  
        }           
    }  
    return instance;

參考
《現(xiàn)代操作系統(tǒng)(第三版)中文版》
《實戰(zhàn)Java高并發(fā)程序設(shè)計》
《Java并發(fā)編程的藝術(shù)》

另外有需要云服務(wù)器可以了解下創(chuàng)新互聯(lián)scvps.cn,海內(nèi)外云服務(wù)器15元起步,三天無理由+7*72小時售后在線,公司持有idc許可證,提供“云服務(wù)器、裸金屬服務(wù)器、高防服務(wù)器、香港服務(wù)器、美國服務(wù)器、虛擬主機、免備案服務(wù)器”等云主機租用服務(wù)以及企業(yè)上云的綜合解決方案,具有“安全穩(wěn)定、簡單易用、服務(wù)可用性高、性價比高”等特點與優(yōu)勢,專為企業(yè)上云打造定制,能夠滿足用戶豐富、多元化的應(yīng)用場景需求。

網(wǎng)頁標(biāo)題:Java并發(fā)之volatile關(guān)鍵字-創(chuàng)新互聯(lián)
分享網(wǎng)址:http://www.muchs.cn/article24/cshhce.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供軟件開發(fā)網(wǎng)站建設(shè)、外貿(mào)建站微信公眾號、App設(shè)計網(wǎng)站收錄

廣告

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

外貿(mào)網(wǎng)站制作