RelativeLayout嵌套RecyclerView可能导致ViewHolder被大量创建

背景:项目当中一个消息列表页面,页面上半部分是一些固定高度的内容。下半部分是消息列表,用RecyclerView实现。最下面有输入框,能打开软键盘。
问题:当消息量大的时候部分手机打开软键盘的时候页面卡顿严重。 经过现象分析发现,

  • 只有在软键盘打开时,消息列表被完全挤掉的手机上才会出现卡顿。
  • 日志输出之后发现RecyclerView为当前Adapter中的所有数据创建了ViewHolder。
  • 进一步跟踪发现这时候的RecyclerView的height的高度已经巨大无比。
【RelativeLayout嵌套RecyclerView可能导致ViewHolder被大量创建】Demo示例如下
布局

Java代码
//adapter recylerView.setAdapter(new RecyclerView.Adapter() { @Override public Holder onCreateViewHolder(ViewGroup parent, int viewType) { TextView textView = new TextView(parent.getContext()); Log.d("aaaa", "onCreateViewHolder "+textView.hashCode()); return new Holder(textView); }@Override public void onBindViewHolder(Holder holder, int position) { Log.d("aaaa", "onBindViewHolder "+position); holder.textView.setText("" + position); }@Override public int getItemCount() { return 10000; } }); //holder class Holder extends RecyclerView.ViewHolder { TextView textView; public Holder(View itemView) { super(itemView); textView = (TextView) itemView; } }

用layout_marginTop模拟页面上部分的固定高度。
当软键盘打开的就会看到onCreateViewHolder被大量调用
总结:这应该是RecyclerView的一个bug。当其被嵌套在RelativeLayout中时如果没有展示空间则其onMeasure得到的高度就会出错。简单的解决办法就是RelativeLayout换成FrameLayout。
至于其内部的具体原因没有去深究。仅此记录

    推荐阅读