在android開發(fā)中Listview是一個(gè)很重要的組件,它以列表的形式根據(jù)數(shù)據(jù)的長(zhǎng)自適應(yīng)展示具體內(nèi)容,用戶可以自由的定義listview每一列的布局,但當(dāng)listview有大量的數(shù)據(jù)需要加載的時(shí)候,會(huì)占據(jù)大量?jī)?nèi)存,影響性能。
本文的重點(diǎn)即是從如下幾個(gè)方面介紹如何對(duì)ListView進(jìn)行優(yōu)化。
1、convertView重用
Android SDK中這樣講:
the old view to reuse, if possible. Note: You should check that this view is non-null and of an appropriate type before using. If it is not possible to convert this view to display the correct data, this method can create a new view
利用好 convertView 來(lái)重用 View,切忌每次 getView() 都新建。ListView 的核心原理就是重用 View,如果重用 view 不改變寬高,重用View可以減少重新分配緩存造成的內(nèi)存頻繁分配/回收;
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:orientation="vertical" >
android:id="@+id/listview"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:cacheColorHint="#00000000" >
ListView的android:layout_height屬性值設(shè)置為"fill_parent"或者''wrap_content"情況不一樣,但是convertView的機(jī)制一樣
如果設(shè)置為fill_parent:屏幕上顯示出的Item的convertview都為空,向下滑動(dòng)新產(chǎn)生的Item的convetview都不為空
如果設(shè)置為wrap_content:只有第一個(gè)Item的convertview為null,其他的不為空
總結(jié):
在初始顯示的時(shí)候,每次顯示一個(gè)item都調(diào)用一次getview方法但是每次調(diào)用的時(shí)候covertview為空(因?yàn)檫€沒有舊的view),當(dāng)顯示完了之后。如果屏幕移動(dòng)了之后,并且導(dǎo)致有些Item(也可以說(shuō)是view)跑到屏幕外面,此時(shí)如果還有新的item需要產(chǎn)生,則這些item顯示時(shí)調(diào)用的getview方法中的convertview參數(shù)就不是null,而是那些移出屏幕的view(舊view),我們所要做的就是將需要顯示的item填充到這些回收的view(舊view)中去,最后注意convertview為null的不僅僅是初始顯示的那些item,還有一些是已經(jīng)開始移入屏幕但是還沒有view被回收的那些item。
2、ViewHolder優(yōu)化
使用ViewHolder的原因是findViewById方法耗時(shí)較大,如果控件個(gè)數(shù)過(guò)多,會(huì)嚴(yán)重影響性能,而使用ViewHolder主要是為了可以省去這個(gè)時(shí)間。通過(guò)setTag,getTag直接獲取View
總結(jié):
view的setTag和getTag方法其實(shí)很簡(jiǎn)單,在實(shí)際編寫代碼的時(shí)候一個(gè)view不僅僅是為了顯示一些字符串、圖片,有時(shí)我們還需要他們攜帶一些其他的數(shù)據(jù)以便我們對(duì)該view的識(shí)別或者其他操作。于是android 的設(shè)計(jì)者們就創(chuàng)造了setTag(Object)方法來(lái)存放一些數(shù)據(jù)和view綁定,我們可以理解為這個(gè)是view 的標(biāo)簽也可以理解為view 作為一個(gè)容器存放了一些數(shù)據(jù)。而這些數(shù)據(jù)我們也可以通過(guò)getTag() 方法來(lái)取出來(lái)。
到這里setTag和getTag大家應(yīng)該已經(jīng)明白了。再回到上面的話題,我們通過(guò)convertview的setTag方法和getTag方法來(lái)將我們要顯示的數(shù)據(jù)來(lái)綁定在convertview上。如果convertview 是第一次展示我們就創(chuàng)建新的Holder對(duì)象與之綁定,并在最后通過(guò)return convertview 返回,去顯示;如果convertview 是回收來(lái)的那么我們就不必創(chuàng)建新的holder對(duì)象,只需要把原來(lái)的綁定的holder取出加上新的數(shù)據(jù)就行了
class ?ViewHolder{
ImageView img;
TextView name;
}
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder = null;
if(convertView==null){
convertView = inflater.inflate(R.layout.list_item, parent, false);
holder.img = (ImageView) convertView.findViewById(R.id.img);
holder.name = (TextView) convertView.findViewById(R.id.name);
holder = new ViewHolder();
convertView.setTag(holder);
}else{
holder = (ViewHolder) convertView.getTag();
}
//設(shè)置holder
holder.img.setImageResource(R.drawable.ic_launcher);
holder.name.setText(list.get(position).partname);
return convertView;
}
3、圖片加載優(yōu)化
如果ListView需要加載顯示網(wǎng)絡(luò)圖片,我們盡量不要在ListView滑動(dòng)的時(shí)候加載圖片,那樣會(huì)使ListView變得卡頓,所以我們需要在監(jiān)聽器里面監(jiān)聽ListView的狀態(tài),如果ListView滑動(dòng)(SCROLL_STATE_TOUCH_SCROLL)或者被猛滑(SCROLL_STATE_FLING)的時(shí)候,停止加載圖片,如果沒有滑動(dòng)(SCROLL_STATE_IDLE),則開始加載圖片。
假如我們要自己實(shí)現(xiàn)應(yīng)該怎么做那,這里提供個(gè)思路
/**
* list滾動(dòng)監(jiān)聽
*/
listView.setOnScrollListener(new OnScrollListener() {
@Override
public void onScrollStateChanged(AbsListView view, int scrollState) {
if (scrollState == OnScrollListener.SCROLL_STATE_IDLE) {//list停止?jié)L動(dòng)時(shí)加載圖片
loadImage(startPos, endPos);// 異步加載圖片 ? ,只加載可以看到的圖片
}
}
@Override
public void onScroll(AbsListView view, int firstVisibleItem,
int visibleItemCount, int totalItemCount) {
//設(shè)置當(dāng)前屏幕顯示的起始pos和結(jié)束pos
startPos = firstVisibleItem;
endPos = firstVisibleItem + visibleItemCount;
if (endPos >= totalItemCount) {
endPos = totalItemCount - 1;
}
}
});
其實(shí)在Universal-Image-loader框架中就存在這個(gè)功能,而且做的很好,完全可以直接拿來(lái)使用,代碼中我們通常這樣設(shè)置:
listView = (ListView) rootView.findViewById(R.id.fragment_user_info_lisiview);
listView.setOnScrollListener(DisplayImageOptionsUtil.getPauseOnScrollListener(this));
listView.setOnItemClickListener(this);
public static PauseOnScrollListener getPauseOnScrollListener(OnScrollListener scrollListener) {
PauseOnScrollListener listener = new PauseOnScrollListener(ImageLoader.getInstance(),
false, true, scrollListener);
return listener;
}
PauseOnScrollListener的第一個(gè)參數(shù)指的是圖片加載對(duì)象ImageLoader,第二個(gè)參數(shù)為pauseOnScroll來(lái)控制是否在滑動(dòng)的過(guò)程中暫停加載圖片,如果需要暫停則傳true,第三個(gè)參數(shù)控制猛的滑動(dòng)界面的時(shí)候圖片是否加載。
打開PauseOnScrollListener的源碼,我們可以看到,在listview滑動(dòng)或者被猛一下滑動(dòng)的時(shí)候,調(diào)用了imageLoader.pause()方法
/**
* Constructor
*
* @param imageLoader ? ?{@linkplain ImageLoader} instance for controlling
* @param pauseOnScroll ?Whether {@linkplain ImageLoader#pause() pause ImageLoader} during touch scrolling
* @param pauseOnFling ? Whether {@linkplain ImageLoader#pause() pause ImageLoader} during fling
* @param customListener Your custom {@link OnScrollListener} for {@linkplain AbsListView list view} which also
* ? ? ? ? ? ? ? ? ? ? ? will be get scroll events
*/
public PauseOnScrollListener(ImageLoader imageLoader, boolean pauseOnScroll, boolean pauseOnFling,
OnScrollListener customListener) {
this.imageLoader = imageLoader;
this.pauseOnScroll = pauseOnScroll;
this.pauseOnFling = pauseOnFling;
externalListener = customListener;
}
@Override
public void onScrollStateChanged(AbsListView view, int scrollState) {
switch (scrollState) {
case OnScrollListener.SCROLL_STATE_IDLE:
imageLoader.resume();
break;
case OnScrollListener.SCROLL_STATE_TOUCH_SCROLL:
if (pauseOnScroll) {
imageLoader.pause();
}
break;
case OnScrollListener.SCROLL_STATE_FLING:
if (pauseOnFling) {
imageLoader.pause();
}
break;
}
if (externalListener != null) {
externalListener.onScrollStateChanged(view, scrollState);
}
}
4、onClickListener處理
當(dāng)ListView的item中有比如button這些子view時(shí),需要對(duì)其設(shè)置onclickListener,通常的寫法是在getView方法中一個(gè)個(gè)設(shè)置,比如
holder.img.setonClickListener(new onClickListenr)...
但是這種寫法每次調(diào)用getView時(shí)都設(shè)置了一個(gè)新的onClick事件,效率很低。高效的寫法可以直接在ViewHolder中設(shè)置一個(gè)position,然后viewHolder implements OnClickListenr:
class ?ViewHolder implements OnClickListener{
int position;
TextView name;
public void setPosition(int position){
this.position = position;
}
@Override
public void onClick(View v) {
switch (v.getId()){
//XXXX
}
}
}
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder = null;
if(convertView==null){
convertView = inflater.inflate(R.layout.list_item, parent, false);
holder = new ViewHolder();
holder.name = (TextView) convertView.findViewById(R.id.name);
holder.name.setOnClickListener(this);
convertView.setTag(holder);
}else{
holder = (ViewHolder) convertView.getTag();
}
//設(shè)置holder
holder.name.setText(list.get(position).partname);
//設(shè)置position
holder.setPosition(position);
return convertView;
}
補(bǔ)充:ListView的listitem里面含有Button ?CheckBox之類的子控件的時(shí)候,子控件會(huì)把Focus搶去,最簡(jiǎn)單有效的解決方法是在ListView的item布局文件根元素中設(shè)置屬性 ?android:descendantFocusability="blocksDescendants"
5、減少Item View的布局層級(jí)
這是所有l(wèi)ayout都必須遵循的,布局層級(jí)過(guò)深會(huì)直接導(dǎo)致View的測(cè)量與繪制浪費(fèi)大量的時(shí)間
6、adapter中的getView方法盡量少使用邏輯
不要在getView方法中做過(guò)于復(fù)雜的邏輯,可以想辦法抽離到別的地方,舉個(gè)例子
優(yōu)化前的getView():
@Override
public View getView(int position, View convertView, ViewGroup paramViewGroup) {
Object current_event = mObjects.get(position);
ViewHolder holder = null;
if (convertView == null) {
holder = new ViewHolder();
convertView = inflater.inflate(R.layout.row_event, null);
holder.ThreeDimension = (ImageView) convertView.findViewById(R.id.ThreeDim);
holder.EventPoster = (ImageView) convertView.findViewById(R.id.EventPoster);
convertView.setTag(holder);
} else {
holder = (ViewHolder) convertView.getTag();
}
//在這里進(jìn)行邏輯判斷,這是有問(wèn)題的
if (doesSomeComplexChecking()) {
holder.ThreeDimention.setVisibility(View.VISIBLE);
} else {
holder.ThreeDimention.setVisibility(View.GONE);
}
// 這是設(shè)置image的參數(shù),每次getView方法執(zhí)行時(shí)都會(huì)執(zhí)行這段代碼,這顯然是有問(wèn)題的
RelativeLayout.LayoutParams imageParams = new RelativeLayout.LayoutParams(measuredwidth, rowHeight);
holder.EventPoster.setLayoutParams(imageParams);
return convertView;
}
優(yōu)化后的getView():
@Override
public View getView(int position, View convertView, ViewGroup paramViewGroup) {
Object object = mObjects.get(position);
ViewHolder holder = null;
if (convertView == null) {
holder = new ViewHolder();
convertView = inflater.inflate(R.layout.row_event, null);
holder.ThreeDimension = (ImageView) convertView.findViewById(R.id.ThreeDim);
holder.EventPoster = (ImageView) convertView.findViewById(R.id.EventPoster);
//設(shè)置參數(shù)提到這里,只有第一次的時(shí)候會(huì)執(zhí)行,之后會(huì)復(fù)用
RelativeLayout.LayoutParams imageParams = new RelativeLayout.LayoutParams(measuredwidth, rowHeight);
holder.EventPoster.setLayoutParams(imageParams);
convertView.setTag(holder);
} else {
holder = (ViewHolder) convertView.getTag();
}
// 我們直接通過(guò)對(duì)象的getter方法代替剛才那些邏輯判斷,那些邏輯判斷放到別的地方去執(zhí)行了
holder.ThreeDimension.setVisibility(object.getVisibility());
return convertView;
}
7、adapter中的getView方法盡量少做耗時(shí)操作
8、adapter中的getView方法避免創(chuàng)建大量對(duì)象
9、將ListView的scrollingCache和animateCache設(shè)置為false
這兩個(gè)屬性,默認(rèn)情況下是開啟的,會(huì)消耗大量的內(nèi)存,因此會(huì)頻繁調(diào)用GC,我們可以手動(dòng)將它關(guān)閉掉(視情況而定)
其它
1、利用好 View Type,例如你的 ListView 中有幾個(gè)類型的 Item,需要給每個(gè)類型創(chuàng)建不同的 View,這樣有利于 ListView 的回收,當(dāng)然類型不能太多
2、善用自定義 View,自定義 View 可以有效的減小 Layout 的層級(jí),而且對(duì)繪制過(guò)程可以很好的控制;
3、盡量能保證 Adapter 的 hasStableIds() 返回 true,這樣在 notifyDataSetChanged() 的時(shí)候,如果 id 不變,ListView 將不會(huì)重新繪制這個(gè) View,達(dá)到優(yōu)化的目的;
4、每個(gè)Item 不能太高,特別是不要超過(guò)屏幕的高度,可以參考 Facebook 的優(yōu)化方法,把特別復(fù)雜的 Item 分解成若干小的 Item
5、ListView 中元素避免半透明
6、盡量開啟硬件加速
7、使用 RecycleView 代替。 ListView 每次更新數(shù)據(jù)都要 notifyDataSetChanged(),有些太暴力了。RecycleView 在性能和可定制性上都有很大的改善,推薦使用。
抽空總結(jié)了一下,同時(shí)加入了一些自己的理解,有問(wèn)題的話,歡迎拍磚! 小伙伴們,求評(píng)論。。。