204.m1-ViewHolder的复用

本文介绍如何在Android ListView中实现多类型的ViewHolder,包括默认ViewHolder和加载更多ViewHolder的具体实现方式,通过BaseAdapter封装共性方法,提高代码复用性和可维护性。

大部分情况下ViewHolder只是一种类型,有时候条目中会有2种holder,比如把ListView中的加载更多当成一个Holder管理的时候,自然有2中Holder,在BaseAdapter中有2个方法

一个是public int getItemViewType(int position) 根据位置 判断当前条目是什么类型,另一个是public int getViewTypeCount()当前ListView 有几种不同的条目类型 ,默认只有一种Holder。

因此在此的实现中,除了正常的ListView的ViewHolder之外,还需要创建一个MoreHolder来加载更多,同时,那两种方法对应的申明应该如下:

下面初始化了2种条目一个是MORE_ITEM加载更多的条目,DEFAULT_ITEM默认的条目

/** 根据位置 判断当前条目是什么类型 */
	@Override
	public int getItemViewType(int position) {
		if (position == datas.size()) { // 当前是最后一个条目,加载更多
			return MORE_ITEM;
		}
		// 如果不是最后一个条目 返回默认类型
		return getInnerItemViewType(position);
	}

	private int getInnerItemViewType(int position) {
		// 返回默认类型
		return DEFAULT_ITEM;
	}

	/** 当前ListView 有几种不同的条目类型 ,默认只有一种Holder*/
	@Override
	public int getViewTypeCount() {
		// TODO Auto-generated method stub
		return super.getViewTypeCount()+ 1; // 2 有两种不同的类型
	}

MoreHolder.java

package com.ldw.market.holder;

import com.ldw.market.R;
import com.ldw.market.utils.UiUtils;

import android.view.View;

/*
 * 加载更多的实现,直接把它当成一个Holder
 */
public class MoreHolder extends BaseHolder<Integer>{

	public static final int HAS_NO_MORE=0;  // 没有额外数据了
	public static final int LOAD_ERROR=1;// 加载失败
	public static final int HAS_MORE=2;//  有额外数据
	
	//MoreHolder的显示
	@Override
	public View initView() {
		View view = View.inflate(UiUtils.getContext(), R.layout.load_more, null);
		return view;
	}

	@Override
	public void refreshView(Integer data) {
		// TODO Auto-generated method stub
		
	}

}

DefaultAdapter.java

package com.ldw.market.Adapter;

import java.util.List;

import com.ldw.market.domain.AppInfo;
import com.ldw.market.holder.BaseHolder;
import com.ldw.market.holder.MoreHolder;

import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;

/*
 * BaseAdapter的共性方法的封装,使用泛型封装
 */
public abstract class DefaultAdapter<Data> extends BaseAdapter{
	protected List<Data> datas;
	private static final int DEFAULT_ITEM = 0;//普通的条目
	private static final int MORE_ITEM = 1;//加载更多的条目
	
	//get方法
	public List<Data> getDatas() {
		return datas;
	}

	//set方法
	public void setDatas(List<Data> datas) {
		this.datas = datas;
	}
	
	//构造函数,初始化数据
	public DefaultAdapter(List<Data> datas) {
		this.datas = datas;
	}

	@Override
	public int getCount() {
		// +1是尾部添加一个加载更多
		return datas.size() + 1;
	}

	
	@Override
	public Object getItem(int position) {
		// TODO Auto-generated method stub
		return datas.get(position);
	}
	
	/** 根据位置 判断当前条目是什么类型 */
	@Override
	public int getItemViewType(int position) {
		if (position == datas.size()) { // 当前是最后一个条目,加载更多
			return MORE_ITEM;
		}
		// 如果不是最后一个条目 返回默认类型
		return getInnerItemViewType(position);
	}

	private int getInnerItemViewType(int position) {
		// 返回默认类型
		return DEFAULT_ITEM;
	}

	/** 当前ListView 有几种不同的条目类型 ,默认只有一种Holder*/
	@Override
	public int getViewTypeCount() {
		// TODO Auto-generated method stub
		return super.getViewTypeCount()+ 1; // 2 有两种不同的类型
	}

	@Override
	public long getItemId(int position) {
		// TODO Auto-generated method stub
		return position;
	}

	@Override
	public View getView(int position, View convertView, ViewGroup parent) {
		//BaseHolder使用泛型以支持多种数据类型的Holder
		BaseHolder holder = null;
		//当前是最后的一个位置显示加载更多MoreHolder
		switch (getItemViewType(position)) {  // 判断当前条目时什么类型
		case MORE_ITEM:	
			//从MoreHolder中获取Holder
			if(convertView==null){
				//加载更多
				holder=getMoreHolder();
			}else{
				//缓冲中获取
				holder=(BaseHolder) convertView.getTag();
			}
			break;
		case DEFAULT_ITEM:
			//从默认中获取Holder
			if (convertView == null) {
				holder = getHolder();
			} else {
				holder = (BaseHolder) convertView.getTag();
			}
			if (position < datas.size()) {
				holder.setData(datas.get(position));
			}
			break;
		}
		return holder.getContentView();
	}


	private BaseHolder getMoreHolder() {
		return new MoreHolder();
	}

	protected abstract BaseHolder<Data> getHolder();

}
HomeFragment.java
package com.ldw.market.fragment;

import java.util.List;

import android.os.Bundle;
import android.text.format.Formatter;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.ImageView;
import android.widget.ListView;
import android.widget.RatingBar;
import android.widget.TextView;

import com.ldw.market.R;
import com.ldw.market.Adapter.DefaultAdapter;
import com.ldw.market.domain.AppInfo;
import com.ldw.market.holder.BaseHolder;
import com.ldw.market.http.HttpHelper;
import com.ldw.market.protocol.HomeProtocol;
import com.ldw.market.utils.UiUtils;
import com.ldw.market.utils.BitmapHelper;
import com.ldw.market.view.BaseListView;
import com.lidroid.xutils.BitmapUtils;
import com.lidroid.xutils.bitmap.PauseOnScrollListener;

/*
 * Home页面,主体的加载的几个页面和服务器请求数据都是在BaseFragment中去实现
 */
public class HomeFragment extends BaseFragment {

	private List<AppInfo> appInfos;

		// 当Fragment挂载的activity创建的时候调用
		@Override
		public void onActivityCreated(Bundle savedInstanceState) {
			super.onActivityCreated(savedInstanceState);
			show();
		}
	
	public View createSuccessView() {
		//使用自定义的ListView,ListView初始化的时候会初始化几个方法
		BaseListView listView = new BaseListView(UiUtils.getContext());
		
		listView.setAdapter(new HomeAdapter(appInfos));
		// 第二个参数 慢慢滑动的时候是否加载图片 false  加载   true 不加载
		//  第三个参数  飞速滑动的时候是否加载图片  true 不加载 
		listView.setOnScrollListener(new PauseOnScrollListener(bitmapUtils, false, true));
		bitmapUtils.configDefaultLoadingImage(R.drawable.ic_default);  // 设置如果图片加载中显示的图片
        bitmapUtils.configDefaultLoadFailedImage(R.drawable.ic_default);// 加载失败显示的图片
		return listView;
	}
	
	private class HomeAdapter extends DefaultAdapter<AppInfo>{

		//构造函数,初始化数据
		public HomeAdapter(List<AppInfo> appInfos) {
			super(appInfos);
			// TODO Auto-generated constructor stub
		}

		@Override
		protected BaseHolder<AppInfo> getHolder() {
			// TODO Auto-generated method stub
			return new ViewHolder();
		}

		
		
	}
	
	static class ViewHolder extends BaseHolder<AppInfo>{
		ImageView item_icon;
		TextView item_title,item_size,item_bottom;
		RatingBar item_rating;

		
		//更新界面
		@Override
		public void refreshView(AppInfo data){
			this.item_title.setText(data.getName());// 设置应用程序的名字
			String size = Formatter.formatFileSize(UiUtils.getContext(), data.getSize());
			this.item_size.setText(size);
			this.item_bottom.setText(data.getDes());
			float stars = data.getStars();
			this.item_rating.setRating(stars); // 设置ratingBar的值
			String iconUrl = data.getIconUrl(); //这个地址不全,需要加上apache地址"iconUrl": "app/com.youyuan.yyhl/icon.jpg",
			//http://127.0.0.1:8090/image?name=app/com.youyuan.yyhl/icon.jpg
			
			
			//第一个参数是显示图片的控件,第二个参数是显示图片的url
			bitmapUtils.display(this.item_icon, HttpHelper.URL+"image?name="+iconUrl);
			// 显示图片的控件
			//bitmapUtils.display(holder.item_icon, HttpHelper.URL+"image?name="+iconUrl);
		}

		@Override
		public View initView() {
			View contentView = View.inflate(UiUtils.getContext(), R.layout.item_app, null);
			this.item_icon = (ImageView) contentView.findViewById(R.id.item_icon);
			this.item_title = (TextView) contentView.findViewById(R.id.item_title);
			this.item_size = (TextView) contentView.findViewById(R.id.item_size);
			this.item_bottom = (TextView) contentView.findViewById(R.id.item_bottom);
			this.item_rating = (RatingBar) contentView.findViewById(R.id.item_rating);
			return contentView;
		}

	}

	@Override
	public LoadResult load() {
		HomeProtocol protocol = new HomeProtocol();
		appInfos = protocol.load(0);
		return checkData(appInfos);
	}
	

	
	
}


<think> 我们讨论的是 `com.chad.library.adapter.base.viewholder` 包下的 `BaseViewHolder` 类。根据引用[1]中出现的异常信息,我们知道 `BaseViewHolder` 是 `BaseRecyclerViewAdapterHelper` 库中的类。在引用[2]中也提到用户使用了这个库。 用户的问题:`com.chad.library.adapter.base.viewholder` 下的 `BaseViewHolder` 没有 `itemview` 属性,无法获取 itemview。 然而,实际上在 `BaseRecyclerViewAdapterHelper` 库的 `BaseViewHolder` 中,`itemView` 是存在的,因为它是 `RecyclerView.ViewHolder` 的子类,而 `RecyclerView.ViewHolder` 本身就有一个 `public final View itemView` 成员。 所以,如果用户遇到无法获取 `itemView` 的情况,可能是以下原因: 1. 用户使用的库版本过旧,在旧版本中可能没有正确暴露这个属性(但通常不会,因为这是RecyclerView的基本要求)。 2. 用户错误地使用了自定义的 ViewHolder 类,并且这个自定义类没有正确继承。 3. 用户在代码中拼写错误(如大小写问题,应为`itemView`而不是`itemview`)。 根据引用[1]中的错误信息:`ClassCastException: com.chad.library.adapter.base.BaseViewHolder cannot be cast to org.redcross.data.common.CommonViewHolder`,这表示用户尝试将 `BaseViewHolder` 强制转换为自定义的 `CommonViewHolder` 类,但这两个类之间没有继承关系,导致转换失败。 因此,用户可能是在自定义 ViewHolder 时出现了问题,而不是 `BaseViewHolder` 没有 `itemView` 属性。 解决方案: 1. 确保自定义的 ViewHolder 继承自 `com.chad.library.adapter.base.viewholder.BaseViewHolder`。 2. 如果不需要自定义 ViewHolder,可以直接使用 `BaseViewHolder`,并通过 `holder.itemView` 来访问。 示例: 在适配器中,通常这样定义: ```java public class MyAdapter extends BaseQuickAdapter<ItemData, BaseViewHolder> { public MyAdapter(int layoutResId, List<ItemData> data) { super(layoutResId, data); } @Override protected void convert(@NonNull BaseViewHolder holder, ItemData item) { // 直接使用 holder.itemView View itemView = holder.itemView; // 或者直接使用 holder 的方法获取子视图 TextView textView = holder.getView(R.id.textView); textView.setText(item.getTitle()); } } ``` 如果用户需要自定义 ViewHolder,应该这样: ```java public class CommonViewHolder extends BaseViewHolder { public CommonViewHolder(View view) { super(view); } } // 在适配器中 public class MyAdapter extends BaseQuickAdapter<ItemData, CommonViewHolder> { public MyAdapter(int layoutResId, List<ItemData> data) { super(layoutResId, data); } @Override protected void convert(@NonNull CommonViewHolder holder, ItemData item) { // 同样,holder.itemView 可用 View itemView = holder.itemView; } @Override protected CommonViewHolder createBaseViewHolder(View view) { return new CommonViewHolder(view); } } ``` 注意:自定义 ViewHolder 必须继承自 `BaseViewHolder`(或者直接继承 `RecyclerView.ViewHolder`,但这样就不能使用 `BaseRecyclerViewAdapterHelper` 提供的便利方法了),并且适配器要正确指定泛型。 如果用户遇到 `com.chad.library.adapter.base.viewholder` 包下的 `BaseViewHolder` 没有 `itemView` 属性,请检查: - 导入的包是否正确:`import com.chad.library.adapter.base.viewholder.BaseViewHolder;` - 是否在自定义 ViewHolder 时覆盖了 `itemView`?正常情况下不会。 另外,引用[3]中的代码展示了如何通过 LayoutInflater 来 inflate 一个视图,这通常在创建 ViewHolder 时使用(在 `onCreateViewHolder` 方法中)。但是,在 `BaseRecyclerViewAdapterHelper` 中,我们通常不需要自己实现 `onCreateViewHolder`,因为适配器已经处理了,我们只需要提供布局ID即可。 总结:用户的问题可能是由于类型转换错误(自定义 ViewHolder 与 BaseViewHolder 不兼容)导致的,而不是 BaseViewHolder 没有 itemView 属性。 因此,建议用户检查自定义 ViewHolder 的继承关系,确保正确继承自 `BaseViewHolder`。 如果用户没有自定义 ViewHolder,那么直接使用 `BaseViewHolder` 并访问 `itemView` 属性即可。 </think>你的问题源于 `com.chad.library.adapter.base.viewholder` 包下的 `BaseViewHolder` 实现差异,以下是完整解决方案: ### 问题根本原因 1. **版本差异**:新版本库(v3.0.4+)将 `itemView` 重命名为 `view` 并优化了访问方式[^2] 2. **类型混淆**:自定义 `CommonViewHolder` 与 `BaseViewHolder` 类型不兼容(引用[1]的 `ClassCastException`) ### 解决方案 #### 方法1:使用正确属性名(推荐) ```java // 新版本的正确访问方式 @Override protected void convert(@NonNull BaseViewHolder holder, ItemData item) { // 获取根视图 View rootView = holder.view; // 替代旧版 holder.itemView rootView.setBackgroundColor(Color.WHITE); // 获取子视图 TextView title = holder.getView(R.id.tv_title); // 使用内置缓存 title.setText(item.getTitle()); } ``` #### 方法2:自定义兼容ViewHolder ```java // 1. 创建兼容ViewHolder public class CommonViewHolder extends BaseViewHolder { public CommonViewHolder(View view) { super(view); } // 添加兼容方法 public View getItemView() { return this.view; // 访问新版本属性 } } // 2. 适配器中使用 public class MyAdapter extends BaseQuickAdapter<ItemData, CommonViewHolder> { @Override protected void convert(@NonNull CommonViewHolder holder, ItemData item) { View root = holder.getItemView(); // 使用自定义方法 // ...操作视图 } } ``` #### 方法3:视图直接访问(应急方案) ```java // 通过布局资源ID直接访问(引用[3]方式) View errorView = LayoutInflater.from(context) .inflate(R.layout.item_layout, recyclerView, false); // 在ViewHolder中 TextView title = errorView.findViewById(R.id.tv_title); ``` ### 版本兼容建议 ```gradle // build.gradle 版本控制 dependencies { // 旧版 (v2.x) implementation 'com.github.CymChad:BaseRecyclerViewAdapterHelper:2.9.50' // 新版 (v3.x+) implementation 'com.github.CymChad:BaseRecyclerViewAdapterHelper:3.0.11' } ``` ### 关键区别说明 | 特性 | v2.x (旧版) | v3.x (新版) | |------|-------------|-------------| | 根视图属性 | `holder.itemView` | `holder.view` | | 子视图获取 | `holder.getView()` | 保持不变 | | 点击事件 | `holder.addOnClickListener()` | `holder.addChildClickViewIds()` | | 性能优化 | 基础缓存 | 增强的SparseArray缓存[^2] | ### 注意事项 1. **避免类型转换错误**: ```java // 错误:类型不兼容(引用[1]) CommonViewHolder holder = (CommonViewHolder) baseHolder; // 正确:统一继承关系 public class CommonViewHolder extends BaseViewHolder { ... } ``` 2. **视图复用问题**: ```java // 避免在convert()外部持有视图引用 // 正确做法在convert()内部实时获取视图 ``` 3. **版本迁移建议**: ```java // 旧版迁移到新版时,全局替换: // holder.itemView → holder.view // holder.getLayoutPosition() → holder.getAbsoluteAdapterPosition() ``` [^1]: 自定义 ViewHolder 需继承正确基类 [^2]: 新版库优化了视图访问方式 [^3]: 直接 inflate 视图作为替代方案
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值