www.MyException.Cn 網友分享於:2015-08-04 瀏覽:0次android
Android之史上最強ListView優化方案緩存
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 viewapp
利用好 convertView 來重用 View,切忌每次 getView() 都新建。ListView 的核心原理就是重用 View,若是重用 view 不改變寬高,重用View能夠減小從新分配緩存形成的內存頻繁分配/回收;框架
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:layout_height="fill_parent" android:orientation="vertical" > <ListView android:id="@+id/listview" android:layout_width="fill_parent" android:layout_height="fill_parent" android:cacheColorHint="#00000000" > </ListView> </LinearLayout>
ListView的android:layout_height屬性值設置爲"fill_parent"或者''wrap_content"狀況不同,可是convertView的機制同樣iview
若是設置爲fill_parent:屏幕上顯示出的Item的convertview都爲空,向下滑動新產生的Item的convetview都不爲空異步
若是設置爲wrap_content:只有第一個Item的convertview爲null,其餘的不爲空ide
總結:佈局
在初始顯示的時候,每次顯示一個item都調用一次getview方法可是每次調用的時候covertview爲空(由於尚未舊的view),當顯示完 了以後。若是屏幕移動了以後,而且致使有些Item(也能夠說是view)跑到屏幕外面,此時若是還有新的item須要產生,則這些item顯示時調用的 getview方法中的convertview參數就不是null,而是那些移出屏幕的view(舊view),咱們所要作的就是將須要顯示的item填 充到這些回收的view(舊view)中去,最後注意convertview爲null的不只僅是初始顯示的那些item,還有一些是已經開始移入屏幕但 是尚未view被回收的那些item。
性能
使用ViewHolder的緣由是findViewById方法耗時較大,若是控件個數過多,會嚴重影響性能,而使用ViewHolder主要是爲了能夠省去這個時間。經過setTag,getTag直接獲取View
總結:
view的setTag和getTag方法其實很簡單,在實際編寫代碼的時候一個view不只僅是爲了顯示一些字符串、圖片,有時咱們還須要他們攜帶一些 其餘的數據以便咱們對該view的識別或者其餘操做。因而android 的設計者們就創造了setTag(Object)方法來存放一些數據和view綁定,咱們能夠理解爲這個是view 的標籤也能夠理解爲view 做爲一個容器存放了一些數據。而這些數據咱們也能夠經過getTag() 方法來取出來。
到這裏setTag和getTag你們應該已經明白了。再回到上面的話題,咱們經過convertview的setTag方法和getTag方法來將咱們 要顯示的數據來綁定在convertview上。若是convertview 是第一次展現咱們就建立新的Holder對象與之綁定,並在最後經過return convertview 返回,去顯示;若是convertview 是回收來的那麼咱們就沒必要建立新的holder對象,只須要把原來的綁定的holder取出加上新的數據就好了
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(); } //設置holder holder.img.setImageResource(R.drawable.ic_launcher); holder.name.setText(list.get(position).partname); return convertView; }
若是ListView須要加載顯示網絡圖片,咱們儘可能不要在ListView滑動的時候加載圖片,那樣會使ListView變得卡頓,因此咱們 須要在監聽器裏面監聽ListView的狀態,若是ListView滑動(SCROLL_STATE_TOUCH_SCROLL)或者被猛滑 (SCROLL_STATE_FLING)的時候,中止加載圖片,若是沒有滑動(SCROLL_STATE_IDLE),則開始加載圖片。
假如咱們要本身實現應該怎麼作那,這裏提供個思路
/** * list滾動監聽 */ listView.setOnScrollListener(new OnScrollListener() { @Override public void onScrollStateChanged(AbsListView view, int scrollState) { if (scrollState == OnScrollListener.SCROLL_STATE_IDLE) {//list中止滾動時加載圖片 loadImage(startPos, endPos);// 異步加載圖片 ,只加載能夠看到的圖片 } } @Override public void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, int totalItemCount) { //設置當前屏幕顯示的起始pos和結束pos startPos = firstVisibleItem; endPos = firstVisibleItem + visibleItemCount; if (endPos >= totalItemCount) { endPos = totalItemCount - 1; } } });
其實在Universal-Image-loader框架中就存在這個功能,並且作的很好,徹底能夠直接拿來使用,代碼中咱們一般這樣設置:
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的第一個參數指的是圖片加載對象ImageLoader,第二個參數爲pauseOnScroll來控制是否在滑動的過程當中暫停加載圖片,若是須要暫停則傳true,第三個參數控制猛的滑動界面的時候圖片是否加載。
打開PauseOnScrollListener的源碼,咱們能夠看到,在listview滑動或者被猛一下滑動的時候,調用了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); } }
當ListView的item中有好比button這些子view時,須要對其設置onclickListener,一般的寫法是在getView方法中一個個設置,好比
holder.img.setonClickListener(new onClickListenr)...
可是這種寫法每次調用getView時都設置了一個新的onClick事件,效率很低。高效的寫法能夠直接在ViewHolder中設置一個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(); } //設置holder holder.name.setText(list.get(position).partname); //設置position holder.setPosition(position); return convertView; }
補充:ListView的listitem裏面含有Button CheckBox之類的子控件的時候,子控件會把Focus搶去,最簡單有效的解決方法是在ListView的item佈局文件根元素中設置屬 性 android:descendantFocusability="blocksDescendants"
這是全部layout都必須遵循的,佈局層級過深會直接致使View的測量與繪製浪費大量的時間
不要在getView方法中作過於複雜的邏輯,能夠想辦法抽離到別的地方,舉個例子
優化前的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(); } //在這裏進行邏輯判斷,這是有問題的 if (doesSomeComplexChecking()) { holder.ThreeDimention.setVisibility(View.VISIBLE); } else { holder.ThreeDimention.setVisibility(View.GONE); } // 這是設置image的參數,每次getView方法執行時都會執行這段代碼,這顯然是有問題的 RelativeLayout.LayoutParams imageParams = new RelativeLayout.LayoutParams(measuredwidth, rowHeight); holder.EventPoster.setLayoutParams(imageParams); return convertView; }
優化後的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); //設置參數提到這裏,只有第一次的時候會執行,以後會複用 RelativeLayout.LayoutParams imageParams = new RelativeLayout.LayoutParams(measuredwidth, rowHeight); holder.EventPoster.setLayoutParams(imageParams); convertView.setTag(holder); } else { holder = (ViewHolder) convertView.getTag(); } // 咱們直接經過對象的getter方法代替剛纔那些邏輯判斷,那些邏輯判斷放到別的地方去執行了 holder.ThreeDimension.setVisibility(object.getVisibility()); return convertView; }
這兩個屬性,默認狀況下是開啓的,會消耗大量的內存,所以會頻繁調用GC,咱們能夠手動將它關閉掉(視狀況而定)
一、利用好 View Type,例如你的 ListView 中有幾個類型的 Item,須要給每一個類型建立不一樣的 View,這樣有利於 ListView 的回收,固然類型不能太多
二、善用自定義 View,自定義 View 能夠有效的減少 Layout 的層級,並且對繪製過程能夠很好的控制;
三、儘可能能保證 Adapter 的 hasStableIds() 返回 true,這樣在 notifyDataSetChanged() 的時候,若是 id 不變,ListView 將不會從新繪製這個 View,達到優化的目的;
四、每一個Item 不能過高,特別是不要超過屏幕的高度,能夠參考 Facebook 的優化方法,把特別複雜的 Item 分解成若干小的 Item
五、ListView 中元素避免半透明
六、儘可能開啓硬件加速
七、使用 RecycleView 代替。 ListView 每次更新數據都要 notifyDataSetChanged(),有些太暴力了。RecycleView 在性能和可定製性上都有很大的改善,推薦使用。