notifyDataSetChanged方法调用后没有作用,必须手动往上滑动一下才显示数据;不是必现的,带有偶然性(始终表现在初始化),时而显示,时而不显示;布局结构是(viewpager+fragment+recyclerview)
目前排除掉一些情况:
- 数据源List没有数据;
- 压根就没调用到notifyDataSetChanged;
- 没有设置layoutmanager
- List的内存地址变掉了,也就是前后设置操作的List不是同一个List;
- 当前notifyDataSetChanged的地方在子线程;
- recyclerview的内存地址前后不一,也就是recyclerview多次findviewbyid,多次实例化,导致堆中控件都不是同一个内存地址(对象);
目前就见识到这6种情况,但是在排除了这6种情况后仍然不行。
先细数一下这6种情况;
数据源List没有数据
这个问题好排查,自己本地打印一下List,或者debug一下就能排查的出来;
压根就没调用到notifyDataSetChanged
这个也好排查,跟上面一样;
没有设置layoutmanager
自己检查一下
List的内存地址变掉了
本地打印一下内存地址,或者debug查看一下内存地址;其实只要你代码写的有条理一点,是不是同一个List是自己能看出来的;
当前notifyDataSetChanged的地方在子线程
这个可以打印当前线程的名字,看是否为main;如果不是可以自己handler切换一下线程,或者你用了线程相关的框架,RxAndroid等,直接调度到主线程执行;
Thread.currentThread().getName()
recyclerview的内存地址前后不一
这个我没遇到过,我看到一些博客上会讲到这个,假设fragment+rv这种方式,有可能使用的不是hide/show方式,用的是replace/add方式,这样每次都要重建视图oncreateview,你的控件可能就是在这里寻址的(findviewbyid);
这种情况下你可以换成hide/show方式,或者定义一些标签或者本身用到的一些变量区别判断第一次oncreateview,避免多次操作;
http://www.jianshu.com/u/f9de259236a3
以上这几种情况我都一一多了排除,发现结果依旧;然后去stackoverflow中找了找类似的情况;
有两种处理方式解决了我的问题;
mRecyclerView.smoothScrollToPosition(0);
第二种方式就是覆写Recyclerview,让他每次都重新layout
private boolean mRequestedLayout = false;
@SuppressLint("WrongCall")
@Override
public void requestLayout() {
super.requestLayout();
// We need to intercept this method because if we don't our children will never update
// Check https://stackoverflow.com/questions/49371866/recyclerview-wont-update-child-until-i-scroll
if (!mRequestedLayout) {
mRequestedLayout = true;
this.post(() -> {
mRequestedLayout = false;
layout(getLeft(), getTop(), getRight(), getBottom());
onLayout(false, getLeft(), getTop(), getRight(), getBottom());
});
}
}
但是具体的原因还是没找到。猜测是线程的问题,和线程导致的问题太像了!