ButterKnife在动态加载布局时出现无法bind问题

场景还原

项目里有一段关于RecyclerView的代码,涉及到动态加载条目项,每个条目除了一小部分内容不同外,基本信息都是一样的。很自然的想到了通过继承来实现差异。<br />
父类:

public abstract class BaseHolder extends RecyclerView.ViewHolder {
    
    @BindView(R.id.container)
    FrameLayout mContainer;

    public BaseHolder(View itemView) {
        super(itemView);
        ButterKnife.bind(this, itemView);
    }
  /*省略所有setter方法*/
}

其中itemview中包含了所有条目都包含的相同基本内容和一个FrameLayout用来实现动态布局加载。<br />

子类:

public class SubHolder extends BaseHolder {

    @BindView(R.id.tv_content)
    TextView mContentView;

    public SubHolder(View itemView) {
        super(itemView);
        Context context = itemView.getContext();
        View root = LayoutInflater.from(context)
                .inflate(R.layout.item_sub_content,mContainer,true);
        //确定绑定情况
        assert null != mContentView;
    }

    public void setContent(String content){
        mContentView.setText(content);
    }
}

在SubHolder中View root就是个性化内容的部分。<br />
这段代码会在运行时抛出:

/*异常是我修改的,上面的代码只是说明问题*/
java.lang.IllegalStateException: Required view 'tv_content' with ID 2131558562 for field 'mContentView' was not found

问题分析

SubHolder构造方法调用super(itemview),进而调用Butterknife.bind(this,itemview),从而从父到子进行View注入,但是这时动态布局root还没有加载mContainer中,所以注入时findViewById()是找不到对于id,导致异常发生。<br />

解决办法

我既不想写单独写findViewById()sssss...(复数加s,/手动滑稽),也不想改变继承结构。<br />
分析现状面临的问题:

  • 基类中的ButterKnife.bind()绑定不了子类中动态加载的View。
  • 子类如果在动态充填(view root = ...)下再次调用ButterKnife.bind(this,root),根据ButterKnife的实现原理(全部再次绑定),基类中的全部注入都会废,因为他们不在root下,那么把root改itemview,就没问题了。
  • 如何让基类中的ButterKnife.bind()放弃检查子类中动态加载的View,而让子类再次调用ButterKnife.bind()来绑定动态加载部分的View。

解决方案:

  • 子类必须重新调用ButterKnife.bind()这是必然的。
  • 子类调用ButterKnife.bind()时必须传入itemView,既包含基类所有view的view root。
  • 通过加入注解@Nullable到子类的动态View中,让基类的ButterKnife.bind()放弃检查它是否绑定上。

修改后的SubHolder:

public class SubHolder extends BaseHolder {

    @BindView(R.id.tv_content)
    @Nullable TextView mContentView;

    public SubHolder(View itemView) {
        super(itemView);
        Context context = itemView.getContext();
        View root = LayoutInflater.from(context)
                .inflate(R.layout.item_sub_content,mContainer,true);
       ButterKnife.bind(this,itemView);
    }

    public void setContent(String content){
        mContentView.setText(content);
    }
}

玉有瑕疵

基类中的view被调用了两次findViewById(),代价很高。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 175,272评论 25 709
  • 一、适用场景 ListViewListview是一个很重要的组件,它以列表的形式根据数据的长自适应展示具体内容,用...
    Geeks_Liu阅读 13,662评论 1 28
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,917评论 18 399
  • 0X0 前言 做过Android开发的猿类很多都知道ButterKnife这么个东西。这个库可以大大的简化我们的代...
    knightingal阅读 4,068评论 1 10
  • 记得之前看过一篇文章,关于牛人思维的,内容几乎都不记得了,只记得里面有个思维是带着问题去睡觉。 因为在睡眠的时候,...
    景景相依阅读 3,693评论 0 0