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

首頁 > 系統(tǒng) > Android > 正文

Android中ListView異步加載圖片錯位、重復、閃爍問題分析及解決方案

2020-04-11 11:27:01
字體:
供稿:網(wǎng)友

Android ListView異步加載圖片錯位、重復、閃爍分析以及解決方案,具體問題分析以及解決方案請看下文。

我們在使用ListView異步加載圖片的時候,在快速滑動或者網(wǎng)絡不好的情況下,會出現(xiàn)圖片錯位、重復、閃爍等問題,其實這些問題總結(jié)起來就是一個問題,我們需要對這些問題進行ListView的優(yōu)化。

比如ListView上有100個Item,一屏只顯示10個Item,我們知道getView()中convertView是用來復用View對象的,因為一個Item的對應一個View對象,而ImageView控件就是View對象通過findViewById()獲得的,而我們在復用View對象時,同時這個ImageView對象也被復用了。比如第11個Item的View復用了第1個Item View對象,那么ImageView就同時被復用了,所以當圖片沒下載出來,這個ImageView(第11個Item)顯示的數(shù)據(jù)就是復用(第1個Item)的數(shù)據(jù)。

1:Item圖片顯示重復

這個顯示重復是指當前行Item顯示了之前某行Item的圖片。

比如ListView滑動到第2行會異步加載某個圖片,但是加載很慢,加載過程中ListView已經(jīng)滑動到了第14行,且滑動過程中該圖片加載結(jié)束。第2行已不在屏幕內(nèi),根據(jù)上面介紹的緩存原理,第2行的View對象可能被第14行復用,這樣我們看到的就是第14行顯示了本該屬于第2行的圖片,造成顯示重復。

2. Item圖片顯示錯亂

這個顯示錯亂是指某行Item顯示了不屬于該行Item的圖片。

跟上面的原因一樣。

3. Item圖片顯示閃爍

上面介紹的另外一種情況,如果第14行圖片又很快加載結(jié)束,所以我們看到第14行先顯示了復用的第2行的圖片,立馬又顯示了自己的圖片進行覆蓋造成閃爍錯亂。

解決方案:

通過上面的分析我們知道了出現(xiàn)錯亂的原因是異步加載及對象被復用造成的,如果每次getView能給對象一個標識,在異步加載完成時比較標識與當前行Item的標識是否一致,一致則顯示,否則不做處理即可。

原理:首先給ImageView設置一個Tag,這個Tag中設置的是圖片的url,然后在加載的時候取得這個url和要加載那position中的url對比,如果不相同就加載,相同就是復用以前的就不加載了。

Android在ListView顯示圖片(重復錯亂閃爍問題)

1、原因分析

ListView item緩存機制:

為了使得性能更優(yōu),ListView會緩存行item(某行對應的View)。

ListView通過adapter的getView函數(shù)獲得每行的item。

滑動過程中

a. 如果某行item已經(jīng)滑出屏幕,若該item不在緩存內(nèi),則put進緩存,否則更新緩存;

b. 獲取滑入屏幕的行item之前會先判斷緩存中是否有可用的item,如果有,做為convertView參數(shù)傳遞給adapter的getView。

這樣,如下的getView寫法就可以充分利用緩存大大提升ListView的性能。即便上萬個行item,最多inflate的次數(shù)為n,

n為一屏最多顯示ListView 行item的個數(shù)。

@Overridepublic View getView ( int position , View convertView , ViewGroup parent ) {  ViewHolder holder ;  if ( convertView == null ) {   convertView = inflater . inflate ( R . layout . list_item , null ) ;   holder = new ViewHolder ( ) ;   ……   convertView . setTag ( holder ) ;  } else {   holder = ( ViewHolder ) convertView . getTag ( ) ;  }}

 這樣提升了性能,但同時也會造成另外一些問題:

a. 行item圖片顯示重復

這個顯示重復是指當前行item顯示了之前某行item的圖片。

比如ListView滑動到第2行會異步加載某個圖片,但是加載很慢,加載過程中l(wèi)istView已經(jīng)滑動到了第14行,且滑動過程中該圖片加載結(jié)束,

第2行已不在屏幕內(nèi),根據(jù)上面介紹的緩存原理,第2行的view可能被第14行復用,這樣我們看到的就是第14行顯示了本該屬于第2行的圖片,

造成顯示重復。

b. 行item圖片顯示錯亂

這個顯示錯亂是指某行item顯示了不屬于該行item的圖片。

比如ListView滑動到第2行會異步加載某個圖片,但是加載很慢,加載過程中l(wèi)istView已經(jīng)滑動到了第14行,第2行已不在屏幕內(nèi),根據(jù)上面介紹的緩存原理,第2行的view可能被第14行復用,第14行顯示了第2行的View,這時之前的圖片加載結(jié)束,就會顯示在第14行,造成錯亂。

c. 行item圖片顯示閃爍

上面b的情況,第14行圖片又很快加載結(jié)束,所以我們看到第14行先顯示了第2行的圖片,立馬又顯示了自己的圖片進行覆蓋造成閃爍錯亂。

2、解決方法

通過上面的分析我們知道了出現(xiàn)錯亂的原因是異步加載及對象被復用造成的,如果每次getView能給對象一個標識,在異步加載完成時比較標識與當前行item的標識是否一致,一致則顯示,否則不做處理即可。

andbase中的實現(xiàn)代碼:

/**  * 顯示這個圖片,解決了列表問題.  * 列表問題:滑動過程中,getView的imageView會重復利用,導致圖片會串位  * @param imageView 顯得的View  * @param url the url  * @return  */  public void display( final ImageView imageView,String url) {   if (AbStrUtil.isEmpty(url)){    if (noImage != null ){     if (loadingView != null ){      loadingView.setVisibility(View.INVISIBLE);      imageView.setVisibility(View.VISIBLE);     }     imageView.setImageDrawable(noImage);    }    return ;   }   //設置下載項   final AbImageDownloadItem item = new AbImageDownloadItem();   //設置顯示的大小   item.width = width;   item.height = height;   //設置為縮放   item.type = type;   item.imageUrl = url;   final String cacheKey = AbImageCache   .getCacheKey(item.imageUrl, item.width, item.height, item.type);   item.bitmap = AbImageCache.getBitmapFromCache(cacheKey);   //if(D) Log.d(TAG, "緩存中獲取的"+cacheKey+":"+item.bitmap);   //設置標記   imageView.setTag(url);   if (item.bitmap == null ){    //先顯示加載中    if (loadingView!= null ){     loadingView.setVisibility(View.VISIBLE);     imageView.setVisibility(View.INVISIBLE);    } else if (loadingImage != null ){     imageView.setImageDrawable(loadingImage);    }    //下載完成后更新界面    item.setListener( new AbImageDownloadListener() {     @Override     public void update(Bitmap bitmap, String imageUrl) {      //未設置加載中的圖片,并且設置了隱藏的View      if (loadingView != null && imageUrl.equals(imageView.getTag())){       loadingView.setVisibility(View.INVISIBLE);       imageView.setVisibility(View.VISIBLE);      }      //要判斷這個imageView的url有變化,如果沒有變化才set,      //有變化就取消,解決列表的重復利用View的問題      if(bitmap!=null&& imageUrl.equals(imageView.getTag())){       if (D) Log.d(TAG, "圖片下載,設置:" +imageUrl);       imageView.setImageBitmap(bitmap);      } else {       if (errorImage != null && imageUrl.equals(imageView.getTag())){        imageView.setImageDrawable(errorImage);       }      }     }    });    if (D) Log.d(TAG, "圖片下載,執(zhí)行:" +url);    mAbImageDownloadPool.execute(item);   } else {    if (loadingView != null ){     loadingView.setVisibility(View.INVISIBLE);     imageView.setVisibility(View.VISIBLE);    }    imageView.setImageBitmap(item.bitmap);   }  }

以上內(nèi)容就是Android中ListView異步加載圖片錯位、重復、閃爍問題分析及解決方案,希望對大家今后的工作和學習有所幫助。

發(fā)表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發(fā)表
主站蜘蛛池模板: 克什克腾旗| 阳信县| 沂水县| 栖霞市| 凤山市| 梧州市| 西盟| 西平县| 广灵县| 泸水县| 嘉义市| 新河县| 遵义市| 新河县| 峨眉山市| 苗栗县| 新田县| 井冈山市| 冷水江市| 如皋市| 交城县| 仲巴县| 佛冈县| 自治县| 习水县| 丰顺县| 财经| 九江县| 北流市| 屯门区| 湘乡市| 唐山市| 武义县| 屯昌县| 广宁县| 平阳县| 赤城县| 松潘县| 额尔古纳市| 张北县| 华安县|