首先先明白一個問題:什么是anr?
anr意思就是應用沒有響應我注。以前存在很多誤解就是"在主線程做了耗時操作"就會引起ANR树叽,現(xiàn)在覺得不完全正確雨膨,耗時操作實際上不一定會導致沒有響應,沒有響應的通俗理解就是:
有人(事件或操作)發(fā)出一個請求,但是主線程沒有對這個人進行回饋(可能是沒時間僚纷,可能是不想理,可能是沒辦法理你)這個就叫做沒有響應拗盒。
這段代碼在onCreate中sleep了10秒怖竭,會出現(xiàn)anr嗎?
答案是可能會陡蝇,也可能不會
當主線程在sleep的時候痊臭,如果UI線程不需要進行操作,也就是說沒有消息會發(fā)送給UI線程并要求UI線程進行處理的時候登夫,那么sleep30秒就不會導致ANR趣兄,因為沒有出現(xiàn)anr的情況,既然沒有人向主線程請求什么東西悼嫉,也就不需要響應艇潭,既然沒有響應了,那么何來說應用無響應,所以就沒有anr存在了蹋凝。
但是當線程在sleep的時候鲁纠,主線程有接收到需要處理的請求的時候,這個時候UI線程正在Sleep鳍寂,根本沒有辦法理你(不想理你)改含,這就符號anr的條件,所以就會出現(xiàn)anr(比如在休眠10秒內(nèi)迄汛,點擊了返回按鈕捍壤。就會出現(xiàn)anr)。
如下圖是安卓11手機上:
雖然主線程休眠了10秒鞍爱,但是在這10秒內(nèi)沒有事件需要處理鹃觉,mButton.setText("first update")更新的操作沒有在主線程sleep的時候被觸發(fā),也就沒有發(fā)生ANR睹逃,
這個同上盗扇,我本以為不會發(fā)生ANR,但現(xiàn)象還是發(fā)生了ANR沉填,猜測安卓11是不是主線程睡眠至少10秒一段時間疗隶。中間會有其他什么事件觸發(fā),導致發(fā)生ANR翼闹,先留個疑問在這斑鼻。
延伸:Thread.sleep問題
由于cpu分配的每個線程的時間片極為短暫(一般為幾十毫秒)所以,cpu通過不停地切換線程執(zhí)行猎荠,這就給程序員造成一種錯覺坚弱,以為是多個線程在同時進行,sleep就是正在執(zhí)行的線程主動讓出cpu法牲,cpu去執(zhí)行其他線程,在sleep指定的時間過后琼掠,此線程重新回到就緒狀態(tài)拒垃,調(diào)用Sleep(1000)函數(shù)并不是說等該線程休眠1秒之后繼續(xù)執(zhí)行它,Thread.Sleep(1000) 意思是在未來的1秒內(nèi)本線程不參與CPU競爭瓷蛙。