先来看看我们之前的项目使用的Adapter的时候是如何开发的呢,先拿一个最简单的Adapter举个例子,这个列表每行只显示一行文字
public class StringAdapter extends BaseListAdapter<String> {
public StringAdapter(Context context, List<String> list) {
super(context, list);
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
if (convertView == null) {
convertView = LayoutInflater.from(mContext).inflate(R.layout.item_simple, null);
}
TextView tv_simple = convertView.findViewById(R.id.tv_simple);
tv_simple.setText(mList.get(position));
return convertView;
}
}
大家可以看到
View convertView = LayoutInflater.from(mContext).inflate(R.layout.item_simple, null);
我们使用这行代码创建了一个View 。就是我们显示的Item的View。
if (convertView == null) {
convertView = LayoutInflater.from(mContext).inflate(R.layout.item_simple, null);
}
我们的代码里是这样写的,做了一个判断,如果不加这个判断直接创建可以不可以呢?答案肯定是不可以啦。那么这行代码的作用是什么呢?
View的复用
举个例子大家就很容易理解了。
假设我的列表有100条数据。比如淘宝的商品列表 搜索之后会返回很多数据100都是少的,那么我们是不是创建100个ItemView每个ItemView 保存一条数据,也就是位每条数据创建一个View呢,也不是不行,知识100View其实已经非常费内存了。这样做之后,很容易出现两个问题,第一个就是卡顿,滑动页面时候容易卡顿,第二个就是直接程序内存溢出。如果1000条数据这样做,基本好手机 都得卡的不要不要的。
所以设计ListView的时候就考虑了这个问题,就是View的复用,虽然我列表有100条数据,但是用户看到的其实可能只有5条,那么我只需要创建5个View就够用户使用了,当用户滑动的时候,最上面的用户已经看不到了,那么我就可以直接拿过来放到最后面直接使用,每个View的布局是一样的,只是内容不一样,那么我只要 对这个View重新赋值就行了,而不是新建一个项目。这样就达到了View复用的目的。
我们之前那个判断就是为了实现这个功能而做的如果不加那个判断,每次滑动的时候都会创建新的View。所以一般情况写ListView都必须加那个判断。效率提升 最少一倍。
那么影响ListView的性能的是什么呢
最影响效率的就是创建View。我们通过上面的代码就可以解决了。
第二个呢就是findViewById()。这个方法的效率其实并不高。对于listview如果页面有非常多的控件,通过这个方法寻找控件,在有的手机都容易产生卡顿,所以我们希望就是每个ItemView我们在获取这个控件之后我们能缓存起来,这样在滑动的时候不需要findViewById()而是直接使用就好了,那么如何缓冲呢这里就有了ViewHolder这个东西。其实他只是一个普通的Java类。没有什么特别的,他里面知识放了Adapter的View的引用,因为每次都保存起来,以后使用就不需要findViewById()而已。
看看代码吧
@Override
public View getView(int position, View convertView, ViewGroup parent) {
if (convertView == null) {
convertView = LayoutInflater.from(mContext).inflate(R.layout.item_simple, null);
}
ViewHolder viewHolder = (ViewHolder) convertView.getTag();
if(viewHolder == null){
viewHolder = new ViewHolder();
viewHolder.tv_simple = convertView.findViewById(R.id.tv_simple);
convertView.setTag(viewHolder);
}
viewHolder.tv_simple.setText(mList.get(position));
return convertView;
}
class ViewHolder{
TextView tv_simple;
}
代码只是改了需要的部分,比之前复杂了很多,但是带来的效率确实实实在在的,如果一个View 使用过那么他就会保持这个ViewHolder,然后我们可以方便的使用了,这样写之后,一般的页面运行就非常快了
ViewHolder的作用就是保存Ttem内的控件的引用,避免每次都需要findView。
知道ViewHolder的作用之后就好理解RecyclerView 里面为什么默认就要求使用ViewHolder了吧,因为效率高嘛。所以ViewHolde变成了一个强制使用的类