Java 并發(fā)編程(3): JAVA 并發(fā)編程基礎

1. JAVA 并發(fā)編程基礎

從啟動一個線程到線程間不同的通信方式.

1.1 線程

線程是系統(tǒng)調(diào)度的最小單位, 擁有各自的計數(shù)器, 堆棧和局部變量等屬性.

1.1.1.1 為什么需要多線程

  • 更多的CPU 核心.
    • 一個線程同一個時刻只能運行在一個CPU 核心上.
  • 更快的響應時間.
  • 更好的編程模型.

1.1.2 線程優(yōu)先級

  • OS 采用時分的形式調(diào)度線程的運行.
    • OS 會分出一個個的時間片, 線程會被分配到若干個時間片.
    • 當線程的時間片用完了就發(fā)生線程調(diào)度, 并等待下次調(diào)度.
  • 線程的優(yōu)先級決定了線程需要被分配的CPU 資源的多少.
    • 頻繁阻塞(休眠或I/O)的線程設置較高的優(yōu)先級.
    • 偏重計算(需要較多CPU 時間的) 設置較低優(yōu)先級, 以防止CPU 被獨占.
  • 但是, OS 和JVM 對線程優(yōu)先級不做任何保證, 程序的正確性不能依賴于線程的優(yōu)先級高低.

1.1.3 線程的狀態(tài)

1.1.4 Daemon 線程

  • 支持型線程: 主要用于程序中后臺調(diào)度及支持性工作.
  • 當一個JVM 中不存在非Daemon 線程的時候, JVM 將會退出.
  • 當JVM 退出時, Daemon 線程中的finally 塊并不一定會被執(zhí)行.
    • Daemon 線程不能依賴于finally 塊來執(zhí)行關閉或清理資源的邏輯.

1.2 啟動和終止線程

1.2.1 構(gòu)造和啟動線程

  • 新構(gòu)造的線程由其parent 線程來進行空間分配, 同時child 繼承了parent 的優(yōu)先級, 是否為Daemon, contextClassLoader 等屬性.
  • 啟動線程時, 最好設置線程名稱, 這樣在使用jstack 分析問題時, 能獲得更多有用的信息.

1.2.2 中斷.

  • 中斷是線程的一個標示位屬性, 它表示一個運行中的線程, 是否被其他線程進行了中斷操作.
  • 線程通過isInterrupted() 來檢查自身是否被中斷, 同時調(diào)用Thread.interrupted() 進行中斷復位.
  • 方法在拋出InterruptedException 之前, JVM 會將該線程的中斷標示位清除, 然后再拋出異常.

1.2.3 過期的suspend(), resume(), stop().

  • 過期的主要原因:
    • 調(diào)用suspend() 后, 線程不會釋放已經(jīng)占有的資源(如鎖), 而是占用著資源進入睡眠狀態(tài), 易引發(fā)死鎖.
    • 調(diào)用stop() 在終結(jié)一個線程時不會爆炸線程的資源正常釋放, 通常是沒有給與線程完成資源釋放的機會, 從而導致程序運行在不確定的狀態(tài)下.
  • 使用新的等待/通知機制來代替它們.

1.2.4 安全地終止線程

  • 通過標識位或中斷操作的方式, 能夠使線程在終止時有機會去清理資源. 而不是武斷地終止線程.
    • 中斷是簡便的線程間交互方式, 適合于取消或停止任務.
    • 同時, 利用一個boolean 變量來控制是否需要停止任務并終止進程.
// class Runner implements Runnable
public void run(){
    while (on && !Thread.currentThread().isInterrupted(){
        // do your job.
    }
}
public void cancel(){
    on = false;
}

// main method
thread.interrupt();
runner.cancel();

1.3 線程間通信

1.3.1 volatile/synchronized 關鍵字

1.3.2 等待/通知機制

  • 生產(chǎn)者/消費者模型, 在功能層面上解耦了How & What.
  • 循環(huán)檢查預期的方案
    • 難以同時保證及時性和降低開銷. 循環(huán)的sleep 時長很難把握.
  • 等待/通知機制依托于同步機制, 其目的是確保等待線程從wait()方法返回時能夠感知到通知線程對變量做出的修改.
    • 使用wait()/notify()/notifyAll()時需要先對調(diào)用對象加鎖.
    • wait(): 放棄鎖并進入對象的WaitQueue中, 進入Waiting狀態(tài).
    • notify(): 將WaitThread 從WaitQueue 移到SynchronizedQueue中, 并將其狀態(tài)變?yōu)锽locked狀態(tài). 在notifyThread 釋放了鎖后, WaitThread 再次獲取到鎖并從wait() 返回并繼續(xù)執(zhí)行.
  • 經(jīng)典范式:
// >>>> 等待方
synchronized(對象){
    while(條件不滿足){
        對象.wait();
    }
    對應的處理邏輯.
}

// >>>> 通知方
synchronized(對象){
    改變條件
    對象.notifyAll();
}

1.3.3 管道輸入/輸出流

  • 與普通的I/O流的不同: 以內(nèi)存為傳輸媒介, 主要用于線程之間的數(shù)據(jù)傳輸.
    • 有面向字節(jié)/字符的PipedInput/OutputStream, PipedReader/Writer.
  • 必須先調(diào)用connect() 進行綁定, 才能進行訪問.

1.3.4 Thread.join() 的使用

  • 每個join線程等待前驅(qū)線程終止后, 才從join()返回.
public final synchronized void join(){
    //條件不滿足, 繼續(xù)等待
    while(isAlive()){
        wait(0);
    }
    //條件符合, 方法返回.
}
  • 當線程終止時,會調(diào)用自身的notifyAll(), 來通知所有等待在該線程對象上的線程.

1.3.5 ThreadLocal 的使用

  • 線程變量: 以ThreadLocal 對象為鍵, 任意對象為值的存儲結(jié)構(gòu).
    • 該結(jié)構(gòu)被附帶在線程上.

1.4 生產(chǎn)者和消費者模式

使用阻塞隊列(容器)做第三方來解耦生產(chǎn)者和消費者, 兩者通過阻塞隊列進行通信.

  • 線程池其實就是一種生產(chǎn)者消費者模式.
    • 線程池的優(yōu)勢是在消費者能夠處理的場景(將要運行的任務數(shù)小于等于線程池的基本任務數(shù)時), 直接就將任務處理掉了.
    • 比原生的生產(chǎn)者先存, 消費者再取的模式更快.
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,119評論 6 531
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,382評論 3 415
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,038評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,853評論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,616評論 6 408
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,112評論 1 323
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,192評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,355評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,869評論 1 334
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 40,727評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,928評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,467評論 5 358
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 44,165評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,570評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,813評論 1 282
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,585評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,892評論 2 372

推薦閱讀更多精彩內(nèi)容

  • 一、進程和線程 進程 進程就是一個執(zhí)行中的程序?qū)嵗總€進程都有自己獨立的一塊內(nèi)存空間,一個進程中可以有多個線程。...
    阿敏其人閱讀 2,620評論 0 13
  • 前言 多線程并發(fā)編程是Java編程中重要的一塊內(nèi)容,也是面試重點覆蓋區(qū)域,所以學好多線程并發(fā)編程對我們來說極其重要...
    嘟爺MD閱讀 7,329評論 21 272
  • 第1章 并發(fā)編程的挑戰(zhàn) 1.1 上下文切換 即便是單核CPU也支持多線程并發(fā),CPU通過給每個線程分配時間片(幾十...
    卑鄙的鹿尤菌閱讀 4,794評論 1 22
  • 1 下公交車的時候 站臺人很多 剛好后面有一輛灑水車開了過來 本以為這下完了 肯定會被灑一身水 突然間 灑水車的車...
    卅槑閱讀 135評論 0 0
  • 今天妹妹心血來潮,要給我表演最好的考級節(jié)目,我趕緊錄了下來,留住這珍貴的一刻。 星期八小鎮(zhèn)玩了...
    擎天柱_6e9a閱讀 131評論 0 0