在Adapter中使用Holder的那些坑

  在使用GridView、ListView时,通常会在Adapter中采用Holder缓存每一项以提高效率,但如果没有用好Holder,这个缓存机制会导致许多意想不到的问题,结合自己的经验特地总结一下,以免今后再犯。

内容错乱

  在Adapter的getView方法中通过position更新每一项的内容,对于根据判断条件给每一项设置属性的情况,每个判断条件下都需要给每一项的每个属性赋值,否则在滑动ListView或GridView时会导致内容错乱,比如下面这段代码是在getView中调用的一个方法,如果在使用Holder时,只在if(position == mSelectPosition)分支下给项背景设置属性,在滑动过程中会导致本来不应该是蓝色背景的项背景变成蓝色了:

private void setFileInfo( FileInfo fileInfo, int position ){
    mFileHolder.mFilePathTxtView.setText( fileInfo.getFilePath( ) );
    mFileHolder.mVoiceTypeTxtView.setText( fileInfo.getVoiceType( ) );
    mFileHolder.mFileItemLayout.setTag(position);

    if(position == mSelectPosition){
        mFileHolder.mFileItemLayout.setBackgroundColor( Color.BLUE );
    }else{
        // 每一项的else分值也需要为项设置背景,否则会在滑动过程中导致内容错乱
        mFileHolder.mFileItemLayout.setBackgroundColor( Color.RED );
    }
}

  这个问题虽然很容易处理,但在开发的过程中经常遇到,比如根据判断条件为每一项设置不同的背景、头像或者文本时,稍有不注意或者偷懒都会导致意想不到的bug,特别是对于每一项需要同时修改多个属性的时候就要更加注意了,一定要保证每个条件分值下都为项的每个视图都设置属性了。

setTag

  在上面那段代码中增加监听项点击事件,由于用到了缓存,所以不能通过getId来获取具体点击了哪一项,通常我们都是通过给每一项的layout设置tag,在监听回调中getTag来获取具体点击了哪一项,比如下面这样:

private void setFileInfo( FileInfo fileInfo, int position ){
    mFileHolder.mFilePathTxtView.setText( fileInfo.getFilePath( ) );
    mFileHolder.mVoiceTypeTxtView.setText( fileInfo.getVoiceType( ) );

    // 为每一项setTag,在onClick方法中通过v.getTag()方法获取点击了哪一项
    mFileHolder.mFileItemLayout.setTag(position);
    mFileHolder.mFileItemLayout.setOnClickListener(new OnClickListener( ) {
        @Override
        public void onClick(View v) {
            int position = ( Integer )v.getTag( );
            mSelectPosition = position;
            if( null != mOnFileListOnClickListener ){
                mOnFileListOnClickListener.onItemClick(mFileInfoList.get(position));
            }

            notifyDataSetChanged( );
        }
    });

    if(position == mSelectPosition){
        mFileHolder.mFileItemLayout.setBackgroundColor( Color.BLUE );
    }else{
        mFileHolder.mFileItemLayout.setBackgroundColor( Color.RED );
    }
}

  如果按上述这种方式编码,在程序运行后点击某一项你会发现程序会挂掉,会报下面的异常:

02-01 15:57:49.340: E/AndroidRuntime(1949): java.lang.ClassCastException: java.lang.Integer cannot be cast to com.uperone.view.FileListAdapter$FileHolder
02-01 15:57:49.340: E/AndroidRuntime(1949): at com.uperone.view.FileListAdapter.getView(FileListAdapter.java:72)

  具体原因也是因为Holder缓存导致的,因为在getView方法中,我是通过为convertView设置tag来做到Holder重用的:

@Override
public View getView(int position, View convertView, ViewGroup parent) {
    if( null == convertView ){
        convertView = mLayoutInflater.inflate( R.layout.list_file_item_layout, null);

        mFileHolder = new FileHolder();
        mFileHolder.mFileItemLayout = ( LinearLayout )convertView.findViewById(R.id.fileItemLayoutId);
        mFileHolder.mFilePathTxtView = ( TextView )convertView.findViewById(R.id.filePathTxtId);
        mFileHolder.mVoiceTypeTxtView = ( TextView )convertView.findViewById(R.id.voiceTypeTxtId);
        mFileHolder.mFileItemLayout.setClickable(true);
        convertView.setTag( mFileHolder );
    }else{
        mFileHolder = ( FileHolder )convertView.getTag( );
    }


    if( !isDataEmpty() ){
        setFileInfo( mFileInfoList.get( position ), position );
    }

    return convertView;
}

  在点击ListView中的某一项时,由于设置每一项内容的setFileInfo方法是在convertView.getTag()后调用的,所以在系统调用getView()方法更新ListView时,convertView取到的tag是mHolder.mFileItemLayout设置的tag,使用了Holder,就不能在每项中有多个视图设置tag和取tag。

  对于上述的异常,通常的解决方式是在ListView实例化的地方,通过setOnItemClickListener方法监听每一项的点击,避免tag冲突的问题。

在KotlinAdapter通常用于Android开发将数据模型转换为列表视图,如RecyclerView或者其他适配器支持的视图。如果你想要在Adapter集成携程(比如携程旅行API),首先你需要做到以下几点: 1. **添加依赖**:确保你的项目已经包含了对携程API的依赖,这可能是一个库或者直接从官方文档获取API客户端。 ```kt dependencies { implementation 'com.ctrip.framework:ctrip-sdk:版本号' } ``` 记得替换`版本号`为实际的携程SDK版本。 2. **初始化服务**:在Adapter的构造函数或某个生命周期管理的地方,初始化携程的服务实例,例如HotelService、FlightService等。 ```kt class YourAdapter(private val hotelService: HotelService) : RecyclerView.Adapter<YourViewHolder>() { //... } ``` 3. **数据请求与处理**:在需要展示数据的地方,通过酒店服务发起请求并解析响应结果。 ```kt override fun onBindViewHolder(holder: YourViewHolder, position: Int) { hotelService.getHotelDetails(position.toLong()) .enqueue(object : Callback<HotelResponse> { override fun onResponse(call: Call<HotelResponse>, response: Response<HotelResponse>) { if (response.isSuccessful) { holder.bindData(response.body()) } else { handleApiError(response) } } override fun onFailure(call: Call<HotelResponse>, t: Throwable) { handleNetworkError(t) } }) } ``` 4. **错误处理**:适当地处理网络错误和API返回的错误,提供用户体验。 5. **遵守携程API规则**:确保你按照携程的开发者协议使用API,并注意请求频率限制和使用场景。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值