Java并發(fā)編程(二):如何保證共享變量的原子性?-創(chuàng)新互聯(lián)

線程安全性是我們在進行 Java 并發(fā)編程的時候必須要先考慮清楚的一個問題。這個類在單線程環(huán)境下是沒有問題的,那么我們就能確保它在多線程并發(fā)的情況下表現(xiàn)出正確的行為嗎?

成都創(chuàng)新互聯(lián)公司成立10余年來,這條路我們正越走越好,積累了技術與客戶資源,形成了良好的口碑。為客戶提供網(wǎng)站設計、成都網(wǎng)站制作、網(wǎng)站策劃、網(wǎng)頁設計、域名注冊、網(wǎng)絡營銷、VI設計、網(wǎng)站改版、漏洞修補等服務。網(wǎng)站是否美觀、功能強大、用戶體驗好、性價比高、打開快等等,這些對于網(wǎng)站建設都非常重要,成都創(chuàng)新互聯(lián)公司通過對建站技術性的掌握、對創(chuàng)意設計的研究為客戶提供一站式互聯(lián)網(wǎng)解決方案,攜手廣大客戶,共同發(fā)展進步。

我這個人,在沒有副業(yè)之前,一心撲在工作上面,所以處理的蠻得心應手,心態(tài)也一直保持的不錯;但有了副業(yè)之后,心態(tài)就變得像坐過山車一樣。副業(yè)收入超過主業(yè)的時候,人特別亢奮,像打了雞血一樣;副業(yè)遲遲打不開局面的時候,人就變得惶惶不可終日。

仿佛我就只能是個單線程,副業(yè)和主業(yè)并行開啟多線程模式的時候,我就變得特別沒有安全感,盡管整體的收入比沒有副業(yè)之前有了很大的改善。

怎么讓我自己變得有安全感,我還沒想清楚(你要是有好的方法,請一定要告訴我)。但怎么讓一個類在多線程的環(huán)境下是安全的,有 3 條法則,讓我來告訴你:

1、不在線程之間共享狀態(tài)變量。
2、將狀態(tài)變量改為不可變。
3、訪問狀態(tài)變量時使用同步。

那你可能要問,狀態(tài)變量是什么?

我們先來看一個沒有狀態(tài)變量的類吧,代碼示例如下。

class Chenmo {
    public void write() {
        System.out.println("我尋了半生的春天,你一笑便是了。");
    }
}

Chenmo 這個類就是無狀態(tài)變量的,它只有一個方法,既沒有成員變量,也沒有類變量。任何訪問它的線程都不會影響另外一個線程的結(jié)果,因為兩個線程之間沒有共享任何的狀態(tài)變量。所以可以下這樣一個結(jié)論:無狀態(tài)變量的類一定是線程安全的。

然后我們再來看一個有狀態(tài)變量的類。假設沉默(Chenmo 類)每寫一行字(write() 方法),就要做一次統(tǒng)計,這樣好找出版社索要稿費。我們?yōu)?Chenmo 類增加一個統(tǒng)計的字段,代碼示例如下。

class Chenmo {
    private long count = 0;
    public void write() {
        System.out.println("我尋了半生的春天,你一笑便是了。");
        count++;
    }
}

Chenmo 類在單線程環(huán)境下是可以準確統(tǒng)計出行數(shù)的,但多線程的環(huán)境下就不行了。因為遞增運算 count++ 可以拆分為三個操作:讀取 count,將 count 加 1,將計算結(jié)果賦值給 count。多線程的時候,這三個操作發(fā)生的時序可能是混亂的,最終統(tǒng)計出來的 count 值就會比預期的值小。

PS:具體的原因可以回顧上一節(jié)《Java 并發(fā)編程(一):摩拳擦掌》。

寫作不易,咱不能虧待了沉默,對不對?那就想點辦法吧。

假定線程 A 正在修改 count 變量,這時候就要防止線程 B 或者線程 C 使用這個變量,從而保證線程 B 或者線程 C 在使用 count 的時候是線程 A 修改過后的狀態(tài)。

怎么防止呢?可以在 write() 方法上加一個 synchronized 關鍵字。代碼示例如下。

class Chenmo {
    private long count = 0;
    public synchronized void write() {
        System.out.println("我尋了半生的春天,你一笑便是了。");
        count++;
    }
}

關鍵字 synchronized 是一種最簡單的同步機制,可以確保同一時刻只有一個線程可以執(zhí)行 write(),也就保證了 count++ 在多線程環(huán)境下是安全的。

在編寫并發(fā)應用程序時,我們必須要保持一種正確的觀念,那就是——首先要確保代碼能夠正確運行,然后再是如何提高代碼的性能。

但眾所周知,synchronized 的代價是昂貴的,多個線程之間訪問 write() 方法是互斥的,線程 B 訪問的時候必須要等待線程 A 訪問結(jié)束,這無法體現(xiàn)出多線程的核心價值。

java.util.concurrent.atomic.AtomicInteger 是一個提供原子操作的 Integer 類,它提供的加減操作是線程安全的。于是我們可以這樣修改 Chenmo 類,代碼示例如下。

class Chenmo {
    private AtomicInteger count = new AtomicInteger(0);
    public void write() {
        System.out.println("我尋了半生的春天,你一笑便是了。");
        count.incrementAndGet();
    }
}

write() 方法不再需要 synchronized 關鍵字保持同步,于是多線程之間就不再需要以互斥的方式來調(diào)用該方法,可以在一定程度上提升統(tǒng)計的效率。

某一天,出版社統(tǒng)計稿費的形式變了,不僅要統(tǒng)計行數(shù),還要統(tǒng)計字數(shù),于是 Chenmo 類就需要再增加一個成員變量了。代碼示例如下。

class Chenmo {
    private AtomicInteger lineCount = new AtomicInteger(0);
    private AtomicInteger wordCount = new AtomicInteger(0);
    public void write() {
        String words = "我這一輩子,走過許多地方的路,行過許多地方的橋,看過許多次的云,喝過許多種類的酒,卻只愛過一個正當年齡的人。";
        System.out.println(words);
        lineCount.incrementAndGet();
       wordCount.addAndGet(words.length());
    }
}

你覺得這段代碼是線程安全的嗎?

結(jié)果顯而易見,這段代碼不是線程安全的。因為 lineCount 和 wordCount 是兩個變量,盡管它們各自是線程安全的,但線程 A 進行 lineCount 加 1 的時候,并不能夠保證線程 B 是在線程 A 執(zhí)行完 wordCount 統(tǒng)計后開始 lineCount 加 1 的。

Java 并發(fā)編程(二):如何保證共享變量的原子性?

該怎么辦呢?方法也很簡單,代碼示例如下。

class Chenmo {
    private int lineCount = 0;
    private int wordCount = 0;
    public void write() {
        String words = "我這一輩子,走過許多地方的路,行過許多地方的橋,看過許多次的云,喝過許多種類的酒,卻只愛過一個正當年齡的人。";
        System.out.println(words);

        synchronized (this) {
            lineCount++;
           wordCount++;
        }
    }
}

對行數(shù)統(tǒng)計(lineCount++)和字數(shù)統(tǒng)計(wordCount++)的代碼進行加鎖,保證這兩行代碼是原子性的。也就是說,線程 B 在進行統(tǒng)計的時候,必須要等待線程 A 統(tǒng)計完之后再開始。

synchronized (lock) {...} 是 Java 提供的一種簡單的內(nèi)置鎖機制,用于保證代碼塊的原子性。線程在進入加鎖的代碼塊之前自動獲取鎖,并且退出代碼塊的時候釋放鎖,可以保證一組語句作為一個不可分割的單元被執(zhí)行。


上一篇:Java 并發(fā)編程(一):簡介

下一篇:如何保證共享變量的原子性?

微信搜索「*沉默王×××免費視頻**」獲取 500G 高質(zhì)量教學視頻(已分門別類)。

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

標題名稱:Java并發(fā)編程(二):如何保證共享變量的原子性?-創(chuàng)新互聯(lián)
新聞來源:http://muchs.cn/article40/dpcoeo.html

成都網(wǎng)站建設公司_創(chuàng)新互聯(lián),為您提供靜態(tài)網(wǎng)站、網(wǎng)站策劃、移動網(wǎng)站建設、面包屑導航、網(wǎng)站改版、全網(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)

小程序開發(fā)