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 在性能和可定制性上都有很大的改善,推荐使用。