您好,登錄后才能下訂單哦!
這期內(nèi)容當(dāng)中小編將會給大家?guī)碛嘘P(guān)在UI主線程中的耗時(shí)操作有什么,文章內(nèi)容豐富且以專業(yè)的角度為大家分析和敘述,閱讀完這篇文章希望大家可以有所收獲。
問題
自Android Ice Cream Sandwich發(fā)布后, 這個(gè)問題就開始在StackOverflow彌散開來:
我的應(yīng)用在Android2.x上運(yùn)行良好,但是在3.x 和4.x系統(tǒng)上總是強(qiáng)退,是什么導(dǎo)致的?
這是一個(gè)很棒的問題,畢竟開發(fā)者總是希望基于舊版本系統(tǒng)開發(fā)的應(yīng)用在新版本的Android系統(tǒng)仍能兼容。在我看來,問題的原因可能多種多樣。 但大多數(shù)時(shí)候,原因非常簡單:你把一個(gè)可能非常耗時(shí)的操作放進(jìn)了UI線程。
什么是UI線程?
應(yīng)用的主UI線程的概念及其重要性是每個(gè)Android開發(fā)者都應(yīng)理解。當(dāng)一個(gè)應(yīng)用啟動,系統(tǒng)會為應(yīng)用創(chuàng)建一個(gè)名為“main”的主線程。這個(gè)主線程(也就是UI主線程)主要負(fù)責(zé)把事件分發(fā)給合適的view或者widget, 因此它非常重要。它也是你的應(yīng)用和應(yīng)用的UI交互的線程。例如,如果你點(diǎn)擊了屏幕上的一個(gè)按鈕,UI線程會把點(diǎn)擊時(shí)間交給view處理,view接到事件后會設(shè)置它的pressed狀態(tài),然后向事件隊(duì)列中發(fā)送一個(gè)invalidate請求。 UI線程會依次讀取隊(duì)列并且告訴view去重繪自己。
除非你的Android應(yīng)用實(shí)現(xiàn)的非常合理,否則這個(gè)單線程模型會使性能變得極低。在極端情況下,如果UI線程負(fù)責(zé)整個(gè)應(yīng)用中的所有操作,進(jìn)行耗時(shí)的操作比如發(fā)送網(wǎng)絡(luò)請求,或者數(shù)據(jù)庫查詢等都會導(dǎo)致用戶界面的阻塞。這些操作在未完成之前,所有的時(shí)間包括繪制和觸屏事件都不會被派發(fā)。從用戶的角度來看,程序似乎是卡死了。
在這些情況下,即時(shí)的反饋相當(dāng)重要。研究表明0.1s是用戶感覺系統(tǒng)是否流暢的臨界值。任何比臨界值更慢的都被認(rèn)為延遲(Miller 1968; Card et al. 1991)。雖然1秒看起來沒什么影響,但在GooglePlay中,即便是十分之一秒也可能是好評和差評的區(qū)別。更糟糕的是,如果UI線程被阻塞5秒以上,用戶會收到“程序未響應(yīng)”(ANR)的提示對話框,并且會強(qiáng)制退出。
為什么Android會使應(yīng)用崩潰
應(yīng)用在2.x系統(tǒng)運(yùn)行良好,在3.0及以上平臺上崩潰的主要原因在于,3.0以上平臺在處理UI線程資源濫用上更加嚴(yán)格。比如說,3.0平臺檢測到UI線程中有網(wǎng)絡(luò)請求時(shí),會拋出NetworkOnMainThreadExceptionwill的異常:
E/AndroidRuntime(673): java.lang.RuntimeException: Unable to start activity ComponentInfo{com.example/com.example.ExampleActivity}: android.os.NetworkOnMainThreadException
Android developer網(wǎng)站文檔中也對此進(jìn)行了很好的解釋:
當(dāng)應(yīng)用試圖在主線程中進(jìn)行網(wǎng)絡(luò)操作,NetworkOnMainThreadException會被拋出。只有在運(yùn)行Honeycomb SDK及更高的版本中會被拋出。更早版本的SDK允許在主事件循環(huán)線程中進(jìn)行網(wǎng)絡(luò)操作,但是非常非常不鼓勵(lì)這么做。
列出一些ICS和Honeycomb不允許在UI線程中進(jìn)行的操作:
打開套接字連接 (i.e. new Socket()).
HTTP 請求 (i.e. HTTPClient and HTTPUrlConnection).
試圖連接遠(yuǎn)程的 MYSQL 數(shù)據(jù)庫.
下載文件 (i.e.Downloader.downloadFile()).
如果你要在UI線程中進(jìn)行某些操作,一定要把它們打包到一個(gè)工作線程中。其中最簡單的方式是使用AsyncTask, 它允許你在你的用戶界面中進(jìn)行一些異步的操作。AsyncTask會把阻塞操作放到工作線程中,并把結(jié)果返回到UI線程,而你不需要處理任何與線程相關(guān)的工作。
上述就是小編為大家分享的在UI主線程中的耗時(shí)操作有什么了,如果剛好有類似的疑惑,不妨參照上述分析進(jìn)行理解。如果想知道更多相關(guān)知識,歡迎關(guān)注億速云行業(yè)資訊頻道。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。