Java—多線程基礎(chǔ)

多線程基礎(chǔ)

基本概念

進(jìn)程

所謂進(jìn)程就是運(yùn)行在操作系統(tǒng)的一個(gè)任務(wù)祭陷,進(jìn)程是計(jì)算機(jī)任務(wù)調(diào)度的一個(gè)單位,操作系統(tǒng)在啟動(dòng)一個(gè)程序的時(shí)候徒扶,會(huì)為其創(chuàng)建一個(gè)進(jìn)程粮彤,JVM就是一個(gè)進(jìn)程。進(jìn)程與進(jìn)程之間是相互隔離的,每個(gè)進(jìn)程都有獨(dú)立的內(nèi)存空間导坟。

計(jì)算機(jī)實(shí)現(xiàn)并發(fā)的原理是:CPU分時(shí)間片屿良,交替執(zhí)行,宏觀并行惫周,微觀串行尘惧。同理,在進(jìn)程的基礎(chǔ)上分出更小的任務(wù)調(diào)度單元就是線程递递,我們所謂的多線程就是一個(gè)進(jìn)程并發(fā)多個(gè)線程喷橙。

線程

在上面我們提到,一個(gè)進(jìn)程可以并發(fā)出多個(gè)線程登舞,而線程就是最小的任務(wù)執(zhí)行單元贰逾,具體來(lái)說(shuō),一個(gè)程序順序執(zhí)行的流程就是一個(gè)線程菠秒,我們常見(jiàn)的main就是一個(gè)線程(主線程)疙剑。

線程的組成

想要擁有一個(gè)線程,有這樣的一些不可或缺的部分践叠,主要有:CPU時(shí)間片言缤,數(shù)據(jù)存儲(chǔ)空間,代碼禁灼。

CPU時(shí)間片都是有操作系統(tǒng)進(jìn)行分配的管挟,數(shù)據(jù)存儲(chǔ)空間就是我們常說(shuō)的堆空間和棧空間匾二,在線程之間哮独,堆空間是多線程共享的,棽烀辏空間是互相獨(dú)立的皮璧,這樣做的好處不僅在于方便,也減少了很多資源的浪費(fèi)分飞。代碼就不做過(guò)多解釋了悴务,沒(méi)有代碼搞個(gè)毛的多線程。

線程的創(chuàng)建和啟動(dòng)
傳統(tǒng)創(chuàng)建線程有兩種方式
  1. 繼承Thread類譬猫,覆蓋run方法

  2. 實(shí)現(xiàn)Runnable接口讯檐,覆蓋run方法

Runnable并不是線程對(duì)象,而是一個(gè)任務(wù)對(duì)象染服。那么Runnable和Thread有什么樣的關(guān)系呢别洪?

通過(guò)查閱API,我們發(fā)現(xiàn)創(chuàng)建一個(gè)線程除了使用Thread的無(wú)參構(gòu)造方法以外有一個(gè)有參構(gòu)造方法是這樣 :Thread(Runnable target)柳刮,通過(guò)這個(gè)方法會(huì)分配一個(gè)新的Thread 對(duì)象挖垛。

其中的參數(shù)是一個(gè)類型為Runnable的target屬性痒钝。

Runnable接口最大的作用就是為非Thread子類的類提供了一種實(shí)現(xiàn)線程的方式,只需要實(shí)現(xiàn)Runnable接口就可以借助Thread創(chuàng)建一個(gè)線程痢毒;另一方面送矩,如果只想重寫(xiě)run方法,不想得到其他的Thread的方法哪替,實(shí)現(xiàn)Runnable是一個(gè)好的選擇栋荸。

JDK1.5

線程池

ExecutorService(線程池 interface)

//通過(guò)工具類中的方法能夠新建一個(gè)線程池,用ExecutorService接受
ExecutorService es = Executors.newFixedThreadPool(2);

Callable對(duì)象

類似于Runnable(描述任務(wù)的interface)凭舶。

//創(chuàng)建一個(gè)Callable的實(shí)現(xiàn)類
Callable<Integer> task1 = new Callable<Integer>(){
  public Integer call() throws Exception{
    int result = 0;
    for(int i=2;i<=100;i+=2){
      result += i;
      Thread.sleep;
    }
      return result;
  }
  
}

//用Future對(duì)象接收f(shuō)ask1的返回值  將任務(wù)提交給線程池
Future<Integer> f = es.submit(task1);
//通過(guò)get方法獲取Future中的值 在這個(gè)時(shí)候主線程主動(dòng)的調(diào)取get  如果分支線程還沒(méi)有結(jié)束晌块,主線程會(huì)在這里阻塞
int result = f1.get();
//關(guān)閉線程池
es.shutdown();

從以上這段代碼我們可以看到很多不一樣的地方,首先在Callable對(duì)象中是可以拋出異常的库快,其次有返回值摸袁,在這個(gè)基礎(chǔ)上也就引出了一個(gè)新的問(wèn)題,如果接收該線程的對(duì)象义屏?JDK1.5中也給出了解決的方法是Future對(duì)象.

啟動(dòng)線程

在這里我們需要明白,上面兩種方式并不會(huì)讓我們得到真正的線程蜂大,只是得到了線程對(duì)象闽铐,只有啟動(dòng)線程,才算得到了真正的線程奶浦。

通過(guò)執(zhí)行start()方法能夠啟動(dòng)一個(gè)線程兄墅,但是啟動(dòng)線程并不是立即執(zhí)行,成功啟動(dòng)的線程會(huì)處于就緒狀態(tài)澳叉,什么時(shí)候執(zhí)行需要等到拿到時(shí)間片之后隙咸。

線程的分類

用戶線程和守護(hù)(Daemon)線程。

守護(hù)線程:守護(hù)線程會(huì)一直運(yùn)行成洗,直到其他非守護(hù)線程都結(jié)束的時(shí)候五督,才會(huì)結(jié)束。有一個(gè)典型的守護(hù)線程就是:垃圾回收線程瓶殃,和虛擬機(jī)共存亡充包,直到虛擬機(jī)中沒(méi)有任何線程的時(shí)候虛擬機(jī)關(guān)閉的時(shí)候才會(huì)終止,簡(jiǎn)單說(shuō)就是虛擬機(jī)在遥椿,它就在基矮,虛擬機(jī)亡便亡。

線程的狀態(tài)

線程的狀態(tài)

上面我們提到過(guò)冠场,一個(gè)線程在啟動(dòng)之后不會(huì)立馬執(zhí)行家浇,而是處于就緒狀態(tài)(Ready),就緒狀態(tài)就是線程的狀態(tài)的一種碴裙,處于這種狀態(tài)的線程意味著一切準(zhǔn)備就緒钢悲, 需要等待系統(tǒng)分配到時(shí)間片灌具。為什么沒(méi)有立馬運(yùn)行呢,因?yàn)橥粫r(shí)間只有一個(gè)線程能夠拿到時(shí)間片運(yùn)行譬巫,新線程啟動(dòng)的時(shí)候讓它啟動(dòng)的線程(主線程)正在運(yùn)行咖楣,只有等主線程結(jié)束,它才有機(jī)會(huì)拿到時(shí)間片運(yùn)行芦昔。

線程的狀態(tài):初始狀態(tài)(New)诱贿,就緒狀態(tài)(Ready),運(yùn)行狀態(tài)(Running)(特別說(shuō)明:在語(yǔ)法的定義中咕缎,就緒狀態(tài)和運(yùn)行狀態(tài)是一個(gè)狀態(tài)Runable)珠十,等待狀態(tài)(Waitering),終止?fàn)顟B(tài)(Terminated)

RUNNABLE)凭豪,等待狀態(tài)(Waitering)焙蹭,終止?fàn)顟B(tài)(Terminated)

初始狀態(tài)(New)

線程對(duì)象被創(chuàng)建出來(lái),便是初始狀態(tài)嫂伞,這時(shí)候線程對(duì)象只是一個(gè)普通的對(duì)象孔厉,并不是一個(gè)線程。

Runable

就緒狀態(tài)(Ready):執(zhí)行start方法之后帖努,進(jìn)入就緒狀態(tài)撰豺,等待被分配到時(shí)間片。

運(yùn)行狀態(tài)(Running):拿到CPU的線程開(kāi)始執(zhí)行拼余。處于運(yùn)行時(shí)間的線程并不是永久的持有CPU直到運(yùn)行結(jié)束污桦,很可能沒(méi)有執(zhí)行完畢時(shí)間片到期,就被收回CPU的使用權(quán)了匙监,之后將會(huì)處于等待狀態(tài)凡橱。

等待狀態(tài)(Waiting)

等待狀態(tài)分為有限期等待和無(wú)限期等待,所謂有限期等待是線程使用sleep方法主動(dòng)進(jìn)入休眠亭姥,有一定的時(shí)間限制稼钩,時(shí)間到期就重新進(jìn)入就緒狀態(tài),再次等待被CPU選中致份。

而無(wú)限期等待就有些不同了变抽,無(wú)限期并不是指永遠(yuǎn)的等待下去,而是指沒(méi)有時(shí)間限制氮块,可能等待一秒也可能很多秒绍载。至于進(jìn)入等待的原因也不盡相同,可能是因?yàn)镃PU時(shí)間片到期滔蝉,也可能是因?yàn)橐粋€(gè)比較耗時(shí)的操作(數(shù)據(jù)庫(kù))击儡,或者主動(dòng)的調(diào)用join方法。

wait和sleep的區(qū)別

wait sleep
wait()方法是Object類里的方法 sleep()是Thread類的static(靜態(tài))的方法
wait()睡眠時(shí)蝠引,釋放對(duì)象鎖 sleep()睡眠時(shí)阳谍,保持對(duì)象鎖蛀柴,仍然占有該鎖
常用于線程間通信 常用于暫停執(zhí)行
wait和notify/notifyAll是成對(duì)出現(xiàn)的, 必須在synchronize塊中被調(diào)用
阻塞狀態(tài)(Blocked)

在我看來(lái),阻塞狀態(tài)實(shí)際上是一種比較特殊的等待狀態(tài)矫夯,處于其他等待狀態(tài)的線程是在等著別的線程執(zhí)行結(jié)束鸽疾,等著拿CPU的使用權(quán);而處于阻塞狀態(tài)的線程等待的不僅僅是CPU的使用權(quán)训貌,主要是鎖標(biāo)記制肮,沒(méi)有拿到鎖標(biāo)記,即便是CPU有空也沒(méi)有辦法執(zhí)行递沪。(關(guān)于鎖見(jiàn)下節(jié):線程同步)

等待和阻塞的區(qū)別

等待 阻塞
已經(jīng)拿到鎖對(duì)象豺鼻,或者說(shuō)不存在拿不到執(zhí)行不了的情況 等待拿到鎖對(duì)象
等待被喚醒 等待拿到鎖對(duì)象
終止線程(Terminated)

已經(jīng)終止的線程會(huì)處于該種狀態(tài)。

總結(jié)

總體上來(lái)說(shuō)款慨,作為一個(gè)線程挺倒霉的儒飒,首先,不會(huì)知道自己什么時(shí)候被選中檩奠;其次在執(zhí)行過(guò)程中隨時(shí)可能被打斷讓出CPU桩了,最后碰到數(shù)據(jù)庫(kù)等耗時(shí)的操作也要讓出CPU去等待,并且就算數(shù)據(jù)準(zhǔn)備好了笆凌, 仍然需要等著被挑選圣猎。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市乞而,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌慢显,老刑警劉巖爪模,帶你破解...
    沈念sama閱讀 218,386評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異荚藻,居然都是意外死亡屋灌,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,142評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門(mén)应狱,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)共郭,“玉大人,你說(shuō)我怎么就攤上這事疾呻〕冢” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 164,704評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵岸蜗,是天一觀的道長(zhǎng)尉咕。 經(jīng)常有香客問(wèn)我,道長(zhǎng)璃岳,這世上最難降的妖魔是什么年缎? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,702評(píng)論 1 294
  • 正文 為了忘掉前任悔捶,我火速辦了婚禮,結(jié)果婚禮上单芜,老公的妹妹穿的比我還像新娘蜕该。我一直安慰自己,他們只是感情好洲鸠,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,716評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布堂淡。 她就那樣靜靜地躺著,像睡著了一般坛怪。 火紅的嫁衣襯著肌膚如雪淤齐。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,573評(píng)論 1 305
  • 那天袜匿,我揣著相機(jī)與錄音更啄,去河邊找鬼。 笑死居灯,一個(gè)胖子當(dāng)著我的面吹牛祭务,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播怪嫌,決...
    沈念sama閱讀 40,314評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼义锥,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了岩灭?” 一聲冷哼從身側(cè)響起拌倍,我...
    開(kāi)封第一講書(shū)人閱讀 39,230評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎噪径,沒(méi)想到半個(gè)月后柱恤,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,680評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡找爱,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,873評(píng)論 3 336
  • 正文 我和宋清朗相戀三年梗顺,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片车摄。...
    茶點(diǎn)故事閱讀 39,991評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡寺谤,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出吮播,到底是詐尸還是另有隱情变屁,我是刑警寧澤,帶...
    沈念sama閱讀 35,706評(píng)論 5 346
  • 正文 年R本政府宣布薄料,位于F島的核電站敞贡,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏摄职。R本人自食惡果不足惜誊役,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,329評(píng)論 3 330
  • 文/蒙蒙 一获列、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧蛔垢,春花似錦击孩、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,910評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至艺玲,卻和暖如春括蝠,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背饭聚。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,038評(píng)論 1 270
  • 我被黑心中介騙來(lái)泰國(guó)打工忌警, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人秒梳。 一個(gè)月前我還...
    沈念sama閱讀 48,158評(píng)論 3 370
  • 正文 我出身青樓法绵,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親酪碘。 傳聞我的和親對(duì)象是個(gè)殘疾皇子朋譬,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,941評(píng)論 2 355

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

  • 寫(xiě)在前面的話: 這篇博客是我從這里“轉(zhuǎn)載”的,為什么轉(zhuǎn)載兩個(gè)字加“”呢兴垦?因?yàn)檫@絕不是簡(jiǎn)單的復(fù)制粘貼徙赢,我花了五六個(gè)小...
    SmartSean閱讀 4,730評(píng)論 12 45
  • 前言 多線程并發(fā)編程是Java編程中重要的一塊內(nèi)容,也是面試重點(diǎn)覆蓋區(qū)域探越,所以學(xué)好多線程并發(fā)編程對(duì)我們來(lái)說(shuō)極其重要...
    嘟爺MD閱讀 7,315評(píng)論 21 272
  • 為什么使用多線程 可以最大限度地利用CPU的空閑時(shí)間來(lái)處理其它任務(wù)犀忱。異步處理不同的任務(wù),提高任務(wù)處理效率扶关。 線程的...
    零度沸騰_yjz閱讀 369評(píng)論 0 4
  • 進(jìn)程:正在執(zhí)行中的程序,其實(shí)是應(yīng)用程序在內(nèi)存中運(yùn)行的那片空間数冬。 線程:進(jìn)程中的一個(gè)執(zhí)行單元节槐,負(fù)責(zé)進(jìn)程中程序的執(zhí)行。...
    七弦桐語(yǔ)閱讀 464評(píng)論 2 7
  • 走了這一趟拐纱,也算是完成了那二三個(gè)月里铜异,默默承諾要做的。希望自己就此心安一段吧秸架! 就這幾天揍庄,時(shí)空轉(zhuǎn)換,季節(jié)也轉(zhuǎn)換了冷...
    lcihnegn0423閱讀 225評(píng)論 0 1