1.一些基本概念
1)線程 進(jìn)程
進(jìn)程則可以說(shuō)是一個(gè)程序,一個(gè)app,而線程是cpu執(zhí)行的最小單元,是一個(gè)順序執(zhí)行流。
當(dāng)一個(gè)程序進(jìn)入內(nèi)存運(yùn)行時(shí),即變成一個(gè)進(jìn)程。進(jìn)程是處于運(yùn)行過(guò)程中的程序,具有一定的獨(dú)立功能。
如果說(shuō)一個(gè)進(jìn)程是一個(gè)公司,那么一個(gè)線程則是一個(gè)員工。
同一個(gè)公司里,不同員工是共享一些資源的,比如飲水機(jī)、打印機(jī)。而不同公司的話,這些資源是獨(dú)立的。
2)并行 并發(fā)
并行指在同一時(shí)刻,有多條指令在多個(gè)處理器上同時(shí)執(zhí)行
并發(fā)是指同一個(gè)時(shí)刻只能有一條指令執(zhí)行,但多個(gè)進(jìn)程指令被快速輪換執(zhí)行。
并行
對(duì)應(yīng)了鳴人的多重影分身,同一時(shí)間,所有的分身都是真的,都可以干活。
而并發(fā)
對(duì)應(yīng)了龜仙人的殘像拳,只有一個(gè)才是真的,只是在高速切換而已。
2.線程的生命周期
線程有5種狀態(tài),分別是new,runnable,running,blocked,dead
.
1)new
新建狀態(tài)
很好記,剛new
了一個(gè)thread()
,它就是new
狀態(tài),新建狀態(tài)。線程剛剛創(chuàng)建完畢,此時(shí)線程和普通的對(duì)象一樣,jvm也剛剛分配內(nèi)存給它,也初始化這個(gè)實(shí)例的成員變量,線程還沒(méi)有開(kāi)始啟動(dòng)。也只有這個(gè)狀態(tài)的線程,可以調(diào)用start()
來(lái)啟動(dòng)線程。
2)runnable
就緒狀態(tài)
當(dāng)一個(gè)線程對(duì)象調(diào)用了start()
以后,這個(gè)線程就處于就緒狀態(tài)了。注意,至于這個(gè)線程什么時(shí)候開(kāi)始運(yùn)行,取決于JVM里線程調(diào)度器的調(diào)度。處于這個(gè)狀態(tài),僅僅表示,這個(gè)線程可以運(yùn)行了。并不是一start()
就馬上就是運(yùn)行狀態(tài)了。
3)running
運(yùn)行狀態(tài)和blocked
阻塞狀態(tài)
如果一個(gè)處于runnable
狀態(tài)的線程獲得了CPU,獲得了寵幸,則開(kāi)始執(zhí)行run()
了,亦即線程執(zhí)行體,我們就說(shuō)這個(gè)線程處于運(yùn)行狀態(tài)。
但一個(gè)線程開(kāi)始運(yùn)行后,它不可能一直處于運(yùn)行狀態(tài)。講道理,它會(huì)被阻塞,以便讓其他線程獲得執(zhí)行的機(jī)會(huì)。不同的平臺(tái)采用的策略不同。對(duì)于采用了搶占式策略的系統(tǒng)而言,阻塞了這個(gè)線程然后選擇下一個(gè)線程的時(shí)候,系統(tǒng)會(huì)考慮線程的優(yōu)先級(jí)。
除了系統(tǒng)主動(dòng)阻塞,還有線程主動(dòng)進(jìn)入阻塞狀態(tài),這一部分我會(huì)在線程控制中講到。
但總之,被阻塞的線程會(huì)在合適的時(shí)候重新進(jìn)入runnable
就緒狀態(tài),注意不是running
運(yùn)行狀態(tài)。所以線程什么時(shí)候重新再執(zhí)行,如上面所說(shuō),是等線程調(diào)度器的調(diào)度的。
4)dead
死亡狀態(tài)
線程有3種死法
1.
run()
或者call()
方法執(zhí)行完成,線程正常結(jié)束
2.線程拋出異常
3.直接調(diào)用該線程的stop()
,但這個(gè)方法不推薦使用,因?yàn)闀?huì)導(dǎo)致死鎖
注意,除非是守護(hù)線程,否則子線程啟動(dòng)起來(lái)后,它就擁有和主線程相同的地位,也就是說(shuō),即便主線程結(jié)束,其他的線程也不受影響,并不會(huì)隨之結(jié)束。這里要注意內(nèi)存泄漏問(wèn)題。
3.線程控制
大量代碼,待續(xù)