?????????????
背景:已經(jīng)有主線(xiàn)程倘屹,為什么還要設(shè)計(jì)一個(gè)子線(xiàn)程?
當(dāng)程序啟動(dòng)以后慢叨,會(huì)被自動(dòng)分配到一個(gè)進(jìn)程和一個(gè)主線(xiàn)程纽匙,主線(xiàn)主要的職責(zé)就是負(fù)責(zé)界面的刷新工作,每16ms就會(huì)收到一個(gè)handler發(fā)來(lái)的來(lái)的message ?一直不停的進(jìn)行刷新界面拍谐,
所以主線(xiàn)程是不能進(jìn)行耗時(shí)的操作(例如復(fù)雜的邏輯運(yùn)算哄辣,超過(guò)百條以上的集合數(shù)據(jù)處理,網(wǎng)絡(luò)請(qǐng)求赠尾,io操作)力穗,如果進(jìn)行了耗時(shí)操作,就會(huì)直接導(dǎo)致主線(xiàn)程阻塞气嫁,也就是常說(shuō)的卡頓(如果界面卡頓当窗,就一定是有其他處理阻塞了界面的正常刷新),甚至系統(tǒng)直接拋出異常寸宵⊙旅妫基于以上的原因爸爸設(shè)計(jì)了子線(xiàn)程,他是和主線(xiàn)程并行的兩條線(xiàn)梯影,并且在合適的時(shí)候合并線(xiàn)程巫员。
界面開(kāi)始100ms以后才會(huì)主線(xiàn)程和子線(xiàn)程,所以理論上來(lái)說(shuō)在oncreate里面直接new Thread其實(shí)沒(méi)什么意義甲棍,但是又可以使用简识。
使用:??
在不指定jvm線(xiàn)程數(shù)請(qǐng)的情況,jvm支持最大線(xiàn)程數(shù)5602
?new Thread(new Runnable() {
????????????@Override
????????????public void run() {
//這里進(jìn)行耗時(shí)操作
????????????}
????????}).start();
[if !supportLists]1感猛、[endif]線(xiàn)程開(kāi)始必須調(diào)用start
[if !supportLists]2七扰、[endif]線(xiàn)程結(jié)束要調(diào)用onstop和interrupt,
[if !supportLists]3陪白、[endif]線(xiàn)程等待和喚醒wait和notify
[if !supportLists]4颈走、[endif]線(xiàn)程等級(jí),所有的線(xiàn)程等級(jí)默認(rèn)都是5咱士,有時(shí)候很多線(xiàn)程進(jìn)行操作的時(shí)候立由,后來(lái)的線(xiàn)程需要優(yōu)先執(zhí)行,這個(gè)時(shí)候就需要給線(xiàn)程增加一個(gè)等級(jí)Thread.setPriority()最大是10序厉,注意這個(gè)線(xiàn)程優(yōu)先級(jí)只是理論上來(lái)說(shuō)會(huì)優(yōu)先執(zhí)行锐膜,并不是高等級(jí)開(kāi)啟以后,直接阻塞其他線(xiàn)程脂矫,也就是最大可能先執(zhí)行他枣耀。
[if !supportLists]5、[endif]線(xiàn)程池Threadpool和AsyncTask簡(jiǎn)單來(lái)講就是集合保存了幾條線(xiàn)程,隨時(shí)拿出來(lái)使用,我們自己一樣可以使用list進(jìn)行實(shí)現(xiàn)
線(xiàn)程切換:怎么從子線(xiàn)程切換到主線(xiàn)程
[if !supportLists]1捞奕、[endif]利用handler和進(jìn)行切換 ?
[if !supportLists]2牺堰、[endif]Activity.runUiThread直接在子線(xiàn)程中使用
[if !supportLists]3、[endif]View.post()及postdelay ?handler也可以直接post
[if !supportLists]4颅围、[endif]AsyncTask使用doinbackground和onpost
[if !supportLists]5伟葫、[endif]Rxjava ??observable
????????????.subscribeOn(Schedulers.io())
????????????.observeOn(AndroidSchedulers.mainThread()).subscribe(observableOnSubscribe)
注意事項(xiàng):
[if !supportLists]1、[endif]線(xiàn)程結(jié)束院促,當(dāng)界面(activity)結(jié)束的以后筏养,一定要將線(xiàn)程結(jié)束,特別是如果線(xiàn)程保有其他對(duì)象的引用常拓,很容易導(dǎo)致內(nèi)存泄漏渐溶。如果線(xiàn)程中使用到循環(huán)(while)的時(shí)候,必須把判斷條件置為false弄抬,茎辐,線(xiàn)程結(jié)束必須在activity調(diào)用在super的ondetory之前
[if !supportLists]2、[endif]線(xiàn)程安全synchronized ??當(dāng)多條線(xiàn)程同時(shí)開(kāi)啟使用同一部分代碼的時(shí)候掂恕,必須給線(xiàn)程增加一個(gè)鎖拖陆,將代碼塊用鎖鎖起來(lái),當(dāng)代碼被使用的時(shí)候懊亡,其他線(xiàn)程自動(dòng)進(jìn)入等待狀態(tài)依啰。原因就是cpu其實(shí)也是單線(xiàn)程操作,每次cpu只能單一功能進(jìn)行操作店枣,在實(shí)用鎖的過(guò)程中盡量不要實(shí)用嵌套鎖速警,每次只實(shí)用一個(gè)鎖來(lái)鎖定一個(gè)對(duì)象,否則容易造成死鎖艰争,鎖定的代碼盡量避免實(shí)用過(guò)于耗時(shí)的操作坏瞄,如果確定耗時(shí)桂对,盡量設(shè)置一個(gè)備用資源甩卓,java有函數(shù)判定線(xiàn)程是否是死鎖狀態(tài)
[if !supportLists]3、[endif]兩個(gè)子線(xiàn)程一樣可以通信蕉斜,做法就是線(xiàn)程1中創(chuàng)建一個(gè)handler逾柿,然后在線(xiàn)程2中使用handler發(fā)消息,當(dāng)handler收到消息宅此,就可以進(jìn)行正常邏輯處理
[if !supportLists]4机错、[endif]