对convertView和ViewHolder的理解

在使用listview和recyclerview的过程中免不了要用到convertview和viewholder,所以理解他们的工作原理还是很有意义的。


首先来看listview的使用,如果要写一个最简单的BaseAdapter的话,那么getView中直接用LayoutInflater根据布局文件生成一个View,然后再获取这个view的子组件设置属性,返回这个view就可以了。但是这个开销显然是不可以接受的,每一次要显示一条列表项的时候都需要执行一次这个过程,根据布局文件生成view。

1.convertView:

convertView是一个作为缓存的view,通过使用这个缓存可以替换掉用Inflater加载组件这一步。

首先做一个实验:

public class MyAdapter extends BaseAdapter {

    private List<String> list;
    private Context context;
    private LayoutInflater inflater;

    public MyAdapter(Context context){
        //this.list=list;
        this.context=context;
        inflater=((Activity)context).getLayoutInflater();
    }

    @Override
    public int getCount() {
        return 100;
    }

    @Override
    public Object getItem(int position) {
        return ""+position;
    }

    @Override
    public long getItemId(int position) {
        return position;
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        if (convertView==null){
            convertView=inflater.inflate(R.layout.item_la,null);
            TextView tv=convertView.findViewById(R.id.tv);
            tv.setText("序号:"+position);
        }
        return convertView;
    }
}
可以看到,convertView为空的时候给他赋值显示position,不为空的时候直接返回显示。
通过效果图发现,前11个列表项(大于一页)正常显示为0到10,从第12个开始再次显示为0到10,随着我缓慢滑动,整个列表总体显示为0-10的循环。但是当我快速滑动时,显示出的一页数字完全乱了。

总结一下,前11个view显示的时候接收的convertView为空,所以要自己给他赋值,也就是装入缓存。11个以后的view接收到convertView就是前面11个一一对应创建的convertView,但是这个顺序快速滑动起来就乱了,所以不能简单认为convertView遵循一个mod11的规律。


2.ViewHolder:

直接看convertView和ViewHolder一起用:

/**书中详细解释该方法*/         
        @Override
        public View getView(finalint position, View convertView, ViewGroup parent) {
             ViewHolder holder;
            //观察convertView随ListView滚动情况             
             Log.v("MyListViewBase", "getView " + position + " " + convertView);
            if (convertView == null) {
                     convertView = mInflater.inflate(R.layout.item,null);
                     holder = new ViewHolder();
                    /**得到各个控件的对象*/                     
                    holder.title = (TextView) convertView.findViewById(R.id.ItemTitle);
                    holder.text = (TextView) convertView.findViewById(R.id.ItemText);
                    holder.bt = (Button) convertView.findViewById(R.id.ItemButton);
                    convertView.setTag(holder);//绑定ViewHolder对象                    
            }
            else{
                    holder = (ViewHolder)convertView.getTag();//取出ViewHolder对象                   
            }
            /**设置TextView显示的内容,即我们存放在动态数组中的数据*/             
            holder.title.setText(getDate().get(position).get("ItemTitle").toString());
            holder.text.setText(getDate().get(position).get("ItemText").toString());
            
            return convertView;
        }
    
    }
	
	/**存放控件*/ 
	publicfinalclass ViewHolder{
	    public TextView title;
	    public TextView text;
	    public Button   bt;
    }


这里ViewHolder的作用其实很明显了,ViewHolder中持有的组件和converView中的组件指向同一个对象,所以这里使用ViewHolder就是替换掉findViewById。



  • 10
    点赞
  • 51
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在 Android 中开发 ListView 或 RecyclerView 时,为了提高列表的滑动流畅度和性能表现,通常会采用 ViewHolder 模式。ViewHolder 是一个用于保存 convertView 中子 View 的对象,可以通过 view.setTag() 方法将其与 convertView 绑定起来。在 getView() 或 onCreateViewHolder() 方法中,如果 convertView 不为 null,则可以直接使用 convertView,并调用 getTag() 方法获取 ViewHolder 对象,避免频繁地调用 findViewById() 方法。这样可以大大减少 UI 线程的工作量,提高应用的性能表现。 具体实现步骤如下: 1. 在 Adapter 中定义 ViewHolder 类,用于保存 convertView 中子 View 的对象。 ```java class ViewHolder { TextView title; ImageView icon; } ``` 2. 在 getView() 或 onCreateViewHolder() 方法中,通过 convertViewViewHolder 实现优化。 ```java @Override public View getView(int position, View convertView, ViewGroup parent) { ViewHolder viewHolder; if (convertView == null) { convertView = LayoutInflater.from(context).inflate(R.layout.item_layout, parent, false); viewHolder = new ViewHolder(); viewHolder.title = convertView.findViewById(R.id.title); viewHolder.icon = convertView.findViewById(R.id.icon); convertView.setTag(viewHolder); } else { viewHolder = (ViewHolder) convertView.getTag(); } // 设置子 View 的数据 viewHolder.title.setText(getItem(position).getTitle()); viewHolder.icon.setImageResource(getItem(position).getIconResId()); return convertView; } ``` 在上述代码中,如果 convertView 不为 null,则通过 convertView.getTag() 方法获取 ViewHolder 对象,否则创建新的 convertViewViewHolder 对象,并将其绑定起来,最后设置子 View 的数据并返回 convertView。 这样,就可以通过 ViewHolder 模式优化列表的性能表现。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值