多線(xiàn)程基礎(chǔ)之常見(jiàn)概念的理解

多線(xiàn)程基礎(chǔ)之常見(jiàn)概念的理解

我這里從線(xiàn)程的關(guān)鍵概念入手精盅,介紹一下多線(xiàn)程,并且結(jié)合代碼說(shuō)下自己的理解谜酒。

線(xiàn)程安全

我們?cè)趺蠢斫膺@個(gè)線(xiàn)程安全叹俏?什么樣的程序是線(xiàn)程安全的?

首先僻族,關(guān)于線(xiàn)程安全的理解粘驰,當(dāng)程序的返回結(jié)果始終與你預(yù)期的一致時(shí),那這段代碼就是線(xiàn)程安全的述么。

那么什么樣的程序是線(xiàn)程安全的呢晴氨?

就我個(gè)人的理解,多線(xiàn)程安全的程序有幾種情況:

  • 程序中不存放共享的數(shù)據(jù)域碉输,多個(gè)線(xiàn)程訪問(wèn)時(shí)根本不會(huì)訪問(wèn)共享的數(shù)據(jù)域籽前,這樣的程序是線(xiàn)程安全的。比如java中的String對(duì)象,它的類(lèi)變量都是final的枝哄,它不可以修改肄梨,就不會(huì)有線(xiàn)程安全的問(wèn)題。
  • 在訪問(wèn)共享的數(shù)據(jù)域時(shí)挠锥,進(jìn)入數(shù)據(jù)域時(shí)众羡,先獲取一把監(jiān)視鎖,保證數(shù)據(jù)在同一時(shí)刻只有一個(gè)線(xiàn)程修改蓖租。這里也很好理解粱侣,比如我們家里都有廁所,一個(gè)人上廁所的時(shí)候蓖宦,通常都會(huì)把門(mén)鎖灼胗ぁ;下一個(gè)人來(lái)了稠茂,只能在外面排隊(duì)等待柠偶,里面的人使用完了,這個(gè)人才能接著使用睬关。

大家都知道java中hashmap是線(xiàn)程不安全的诱担,下面例子去說(shuō)明hashmap的線(xiàn)程不安全。

package com.allen.dayup.高并發(fā)程序設(shè)計(jì).chap1;

import java.util.HashMap;
import java.util.Map;

/**
 * @Auther: allen
 * @Date: 2020-03-20 08:45
 * @Description:
 */
public class HashMapMutiThread {

    static Map<String,String> map = new HashMap<>();

    public static class AddThread extends Thread {

        private int start = 0;

        public AddThread(String name,int start) {
            super(name);
            this.start = start;
        }

        @Override
        public void run() {
            for (int i = start; i < 100000; i+=2) {
                map.put(String.valueOf(i), String.valueOf(i));
            }
        }
    }

    public static void main(String[] args) throws InterruptedException {
        AddThread addThread = new AddThread("Add_1", 0);
        AddThread addThread2 = new AddThread("Add_2", 1);
        addThread.start();
        addThread2.start();

        addThread.join();
        addThread2.join();

        System.out.println("Map的size:" + map.size());
    }
}

程序運(yùn)行完后电爹,可能有兩種結(jié)果(JDK8):

  1. map的size剛好等于100000蔫仙,程序運(yùn)行正常。
  2. map的size小于100000丐箩,這是因?yàn)閙ap在擴(kuò)容過(guò)程中摇邦,并沒(méi)有加鎖的,導(dǎo)致另一個(gè)線(xiàn)程成訪問(wèn)到了不一致的內(nèi)部狀態(tài)雏蛮。

可見(jiàn)性和不可見(jiàn)性

討論可見(jiàn)性之前,我們要稍微提一下計(jì)算機(jī)的內(nèi)存模型阱州,計(jì)算機(jī)為了更好發(fā)揮cpu的性能挑秉,內(nèi)部有三級(jí)緩存結(jié)構(gòu)。cpu上的計(jì)算結(jié)果并不是實(shí)時(shí)刷新到主存的苔货。對(duì)于java的普通變量來(lái)說(shuō)犀概,每個(gè)線(xiàn)程都有自己一塊空間存儲(chǔ)變量,一個(gè)線(xiàn)程修改了變量夜惭,并不會(huì)實(shí)時(shí)把變量刷到主存時(shí)姻灶;在變量沒(méi)有刷新的時(shí)間段內(nèi),對(duì)于另外一個(gè)線(xiàn)程诈茧,變量的修改就是不可見(jiàn)的产喉。

在java可以用volatile關(guān)鍵字來(lái)保證變量的可見(jiàn)性。

下面通過(guò)一個(gè)例子來(lái)說(shuō)明線(xiàn)程的可見(jiàn)性。

package com.allen.dayup.高并發(fā)程序設(shè)計(jì).chap1;

/**
 * @Auther: allen
 * @Date: 2020-03-19 22:15
 * @Description:
 */
public class NoVisibility {
    //當(dāng)ready為非volatile變量時(shí)曾沈,程序不能正常退出这嚣,因?yàn)閞eader線(xiàn)程不能看到主線(xiàn)程對(duì)變量的修改
    private static volatile boolean ready = false;
    private static int number;

    private static class Reader extends Thread{

        @Override
        public void run() {
            while ( !ready ){

            }
            System.out.println(number);
        }
    }

    public static void main(String[] args) throws InterruptedException{
        Reader reader = new Reader();
        reader.start();
        number = 42;
        ready = true;
    }
}

如果ready是非volatile變量時(shí),程序不能正常退出塞俱。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末姐帚,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子障涯,更是在濱河造成了極大的恐慌罐旗,老刑警劉巖,帶你破解...
    沈念sama閱讀 219,039評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件唯蝶,死亡現(xiàn)場(chǎng)離奇詭異九秀,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)生棍,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,426評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門(mén)颤霎,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人涂滴,你說(shuō)我怎么就攤上這事友酱。” “怎么了柔纵?”我有些...
    開(kāi)封第一講書(shū)人閱讀 165,417評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵缔杉,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我搁料,道長(zhǎng)或详,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,868評(píng)論 1 295
  • 正文 為了忘掉前任郭计,我火速辦了婚禮霸琴,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘昭伸。我一直安慰自己梧乘,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,892評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布庐杨。 她就那樣靜靜地躺著选调,像睡著了一般。 火紅的嫁衣襯著肌膚如雪灵份。 梳的紋絲不亂的頭發(fā)上仁堪,一...
    開(kāi)封第一講書(shū)人閱讀 51,692評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音填渠,去河邊找鬼弦聂。 笑死鸟辅,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的横浑。 我是一名探鬼主播剔桨,決...
    沈念sama閱讀 40,416評(píng)論 3 419
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼徙融!你這毒婦竟也來(lái)了洒缀?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 39,326評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤欺冀,失蹤者是張志新(化名)和其女友劉穎树绩,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體隐轩,經(jīng)...
    沈念sama閱讀 45,782評(píng)論 1 316
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡饺饭,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,957評(píng)論 3 337
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了职车。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片瘫俊。...
    茶點(diǎn)故事閱讀 40,102評(píng)論 1 350
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖悴灵,靈堂內(nèi)的尸體忽然破棺而出扛芽,到底是詐尸還是另有隱情,我是刑警寧澤积瞒,帶...
    沈念sama閱讀 35,790評(píng)論 5 346
  • 正文 年R本政府宣布川尖,位于F島的核電站,受9級(jí)特大地震影響茫孔,放射性物質(zhì)發(fā)生泄漏叮喳。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,442評(píng)論 3 331
  • 文/蒙蒙 一缰贝、第九天 我趴在偏房一處隱蔽的房頂上張望馍悟。 院中可真熱鬧,春花似錦剩晴、人聲如沸锣咒。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,996評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)宠哄。三九已至壹将,卻和暖如春嗤攻,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背诽俯。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,113評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工妇菱, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留承粤,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,332評(píng)論 3 373
  • 正文 我出身青樓闯团,卻偏偏與公主長(zhǎng)得像辛臊,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子房交,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,044評(píng)論 2 355