背景:项目当中一个消息列表页面,页面上半部分是一些固定高度的内容。下半部分是消息列表,用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。
至于其内部的具体原因没有去深究。仅此记录
网友评论