今天在写首页的时候遇到一个问题,采用的是Viewpage+TabLayout+Fragment的布局,然后发现后台添加数据的时候或者更改tablayout的数据位置的,viewpage的数据显示不对应,请教大佬终于找到了解决方案
ViewPager与TabLayout的初始化:
private void initData(){
mTabFragmentAdapter = new HomeFragmentPagerAdapter(getChildFragmentManager());
mViewPager.setAdapter(mTabFragmentAdapter);
mViewPager.setOffscreenPageLimit(0);
mViewPager.addOnPageChangeListener(
new TabLayout.TabLayoutOnPageChangeListener(mTabLayout));
mTabLayout.setupWithViewPager(mViewPager);
mViewPager.setIsScroll(false, false);
}
2.刷新数据FragmentPagerAdapter的notifyDataSetChanged()

3,FragmentPagerAdapter主要设置了
public class HomeFragmentPagerAdapter extends FragmentPagerAdapter {
private List<RecommendBean> mTitles = new ArrayList<>();
public void setDataAndType(List<RecommendBean> titles) {
mTitles.clear();
mTitles.addAll(titles);
}
@SuppressWarnings("deprecation")
public HomeFragmentPagerAdapter(FragmentManager fm) {
super(fm);
}
@NotNull
@Override
public Fragment getItem(int position) {
int id = mTitles.get(position).getId();
Log.d("test","id = "+id);
return HomeGoodsFragment.newInstance(id);
}
@Override
public int getCount() {
return this.mTitles == null ? 0 : this.mTitles.size();
}
@Nullable
@Override
public CharSequence getPageTitle(int position) {
return mTitles.get(position).getName();
}
@Override
public void destroyItem(@NonNull ViewGroup container, int position, @NonNull Object object) {
// super.destroyItem(container, position, object);
}
@Override
public void restoreState(Parcelable state, ClassLoader loader) {
// super.restoreState(state, loader);
}
@Override
public int getItemPosition(@NotNull Object object) {
Log.d("test","刷新数据");
((HomeGoodsFragment) object).refresh();
return POSITION_NONE;
}
@Override
public long getItemId(int position) {
return mTitles.get(position).getId();
}
}
我们发现每次创建fragment时候,FragmentManager都会通过findFragmentByTag去缓存中查找,是否存在指定tagName的Fragment,有就复用,没有就调用getItem()创建新的。而这里的tagName就是通过getItemId()方法生成的。
原来FragmentPagerAdapter里在根据getItemId(int position)来判断当前position里Fragment是否存在,如果存在,则不会创建亦不会更新,那么要让FragmentPagerAdapter的更新生效,那在getItemId(int)里根据数据返回一个唯一的数据ID,当FragmentPagerAdapter更新时,数据ID改变了,那么Fragment就会调用getItem(int)去获取新Fragment,达到更新效果 好了,我们的解决方案就是通过重新getItemId()方法,返回唯一的id。

mTabItemIds我是使用的每个tab的id集合。然后返回不同tabId的hashcode作为唯一id。 同时还要重写getItemPosition()方法,通知刷新位置变化了。

|