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

背景:项目当中一个消息列表页面,页面上半部分是一些固定高度的内容。下半部分是消息列表,用RecyclerView实现。最下面有输入框,能打开软键盘。

问题:当消息量大的时候部分手机打开软键盘的时候页面卡顿严重。

经过现象分析发现,

  • 只有在软键盘打开时,消息列表被完全挤掉的手机上才会出现卡顿。
  • 日志输出之后发现RecyclerView为当前Adapter中的所有数据创建了ViewHolder。
  • 进一步跟踪发现这时候的RecyclerView的height的高度已经巨大无比。

Demo示例如下
布局

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/activity_recyle_view"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context="com.example.mozat.myapplication.RecyleViewActivity">

    <android.support.v7.widget.RecyclerView
        android:id="@+id/recylerView"
        android:layout_marginTop="350dp"
        android:layout_width="match_parent"
        android:layout_height="match_parent"></android.support.v7.widget.RecyclerView>

    <EditText
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_alignParentBottom="true"/>
</RelativeLayout>

Java代码

//adapter
recylerView.setAdapter(new RecyclerView.Adapter<Holder>() {
    @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。

至于其内部的具体原因没有去深究。仅此记录

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

推荐阅读更多精彩内容