国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 系統 > Android > 正文

Android異步回調中的UI同步性問題分析

2019-12-12 06:17:22
字體:
來源:轉載
供稿:網友

Android程序編碼過程中,回調無處不在。從最常見的Activity生命周期回調開始,到BroadcastReceiver、Service以及Sqlite等。Activity、BroadcastReceiver和Service這些基本組件的回調路徑和過程也就是通常意義上所謂的“生命周期”。同時,在處理具體的業務邏輯時,常常設計到不同線程之間的通信,如下載圖片完成后通知 UI線程更新UI,凡此類場景,無論使用哪一種具體的線程間通信方式(Handler/Message、Handler/post、基于接口的回調、基于多對多的觀察者模式如EventBus等),其本質上都是基于“回調”。在實際編碼過程中,凡涉及到不同線程之間的通信,本質上更是屬于“異步回調”。當需要在“異步回調”中修改UI時,此時需要特別注意UI同步性問題。

為了便于問題的闡述,在此先對“Android異步回調UI同步性問題”進行如下界定:當異步回調執行時(稱之為“異步回調執行點”),當前UI界面上的元素與最初生成此異步回調的調用器開始執行時(稱之為“異步回調生成點”)的UI元素已經存在不一致,不一致不僅包括UI元素可能的界面變化、可能的內容變化,也包括“異步回調執行點”和“異步回調生成點”時的UI元素中的某一特性的表征量(如某一具有表征當前UI元素的字段值)相關,即使UI元素界面和內容都尚未發生變化。

編碼過程中,“Android異步回調UI同步性問題”經常存在,有時候稍不注意會產生一些看起來難以理解的bug,并由于異步特性的存在,此類bug還具有一定的隨機性。有時候由于一些需求的復雜性,此類bug隱蔽性很強,也容易被忽略。至少到目前為止,在實際開發中,本人遇到此類問題已有數個。

純文字的描述可能不太好理解,下面以一個很常用的Android-Universal-Image-Loader為例,簡單舉例一個潛在存在的“Android異步回調UI同步性問題”。

ListView Item View中有ImageView,通過Android-Universal-Image-Loader去加載顯示,圖片加載完成后需要做一些邏輯處理(如隱藏圖片加載進度條等..),通常代碼如下:

ImageLoader.getInstance().loadImage(imageUrl, new ImageLoadingListener() {         @Override public void onLoadingComplete(String imageUri, View view, Bitmap loadedImage) {  if (loadedImage != null) {   imageView.setImageBitmap(loadedImage);   // 其他業務邏輯處理..  } } @Override public void onLoadingStarted(String imageUri, View view) {   } @Override public void onLoadingCancelled(String arg0, View arg1) {   } @Override public void onLoadingFailed(String arg0, View arg1, FailReason arg2) {   }});

初看上去,代碼邏輯好像也沒什么問題,網上大部分人也是這么寫的。當較慢滑動ListView時,或在平時正常使用時,也沒有什么問題。但是此處的代碼邏輯真的嚴密嗎?

ListView的getView復用特性,大家也都熟知。對于之前遇到的“圖片錯位/先顯示之前的圖片后再被正確的圖片覆蓋掉”,此類現象也都知道如何解決(在getView邏輯開始處理處將ImageView設置成最先的默認圖片,其他UI元素類似處理),基本上也不會再有“圖片錯位/先顯示之前的圖片后再被正確的圖片覆蓋掉”這類現象了。實際上,當網速條件一般,且loadImage大致與上述代碼所示,在ListView中快速滑動列表,幾屏后,不出意外,會發現“圖片錯位/先顯示之前的圖片后再被正確的圖片覆蓋掉”此問題依然存在。

此時問題出現的原因不在于getView本身,因為getView邏輯開始時已經將ImageView重置為默認圖片,而在于“Android異步回調UI同步性問題”。由于ViewHolder的不斷復用,網速一般時快速滑動幾屏后,onLoadingComplete的異步回調執行時與當前UI元素已經存在不一致,簡單點理解,ImageView被復用了ImageView position 0,ImageView position 11, ImageView position 21,此時滑動停止,onLoadingComplete的異步回調執行時ImageView已經是最后一次的ImageView position 21,而onLoadingComplete的異步回調可能被執行數次(ImageView position 0,ImageView position 11, ImageView position 21,且順序還取決于異步中的具體處理和網絡環境等),于是問題發生了。

解決方案:
抓住”UI元素中的某一特性的表征量“,在異步回調中通過比較“異步回調生成點”和“異步回調執行點”此特征變量的值直接作出邏輯上的處理。

public class HardRefSimpleImageLoadingListener implements ImageLoadingListener { public int identifier; public HardRefSimpleImageLoadingListener() { } public HardRefSimpleImageLoadingListener(int identifier) {  this.identifier = identifier; } @Override public void onLoadingCancelled(String arg0, View arg1) { } @Override public void onLoadingComplete(String arg0, View arg1, Bitmap arg2) { } @Override public void onLoadingFailed(String arg0, View arg1, FailReason arg2) { } @Override public void onLoadingStarted(String arg0, View view) {  }}ImageLoader.getInstance().loadImage(imageUrl, new HardRefSimpleImageLoadingListener(did) { @Override public void onLoadingComplete(String imageUri, View view, Bitmap loadedImage) {  if (loadedImage != null) {   if (identifier != did) {    return;   }   imageView.setImageBitmap(loadedImage);   // 其他業務邏輯處理..  } }});


總之,凡此類“Android異步回調UI同步性問題”,最好都通過比較“異步回調生成點”“異步回調執行點”特征變量的值去針對性的做邏輯處理,以免出現不必要的Bug,是非常必要且有效的手段。

 原文地址:http://www.cnblogs.com/lwbqqyumidi/p/4110377.html

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持武林網。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 莆田市| 景德镇市| 嵊州市| 抚顺市| 安义县| 崇信县| 湘潭县| 慈利县| 平顶山市| 海丰县| 永定县| 白沙| 平江县| 丹凤县| 通辽市| 桐乡市| 梨树县| 嘉荫县| 巴彦县| 三河市| 长宁区| 临汾市| 平凉市| 苏州市| 会泽县| 兰考县| 榆社县| 宁乡县| 逊克县| 治县。| 玉田县| 铁岭县| 泰州市| 石城县| 平远县| 长春市| 纳雍县| 徐汇区| 万州区| 黄骅市| 旺苍县|