项目中一个页面中包含一个ViewPager,适配器为FragmentPagerAdapter的实现类,当页面选中时,会请求本页数据,请求结束刷新对应下标fragment的数据,伪代码如下:
private int currentPosition;//viewPager 当前选中下标
protected void onCreate(Bundle savedInstanceState) {
...
List<Fragment> fragmentList = new ArrayList();
for(int i = 0;i< mData.size();i++){
fragmentList.add(MyFragment.newInstance(i));
}
...
viewPager.setAdapter(new FragmentPagerAdapter(getSupportFragmentManager()) {
@Override
public Fragment getItem(int i) {
return fragmentList.get(i);
}
@Override
public int getCount() {
return fragmentList.size();
}
});
viewPager.addOnPageChangeListener(new ViewPager.OnPageChangeListener() {
@Override
public void onPageScrolled(int i, float v, int i1) {
}
@Override
public void onPageSelected(int i) {
//请求数据
currentPosition = i;
requestData();
}
@Override
public void onPageScrollStateChanged(int i) {
}
});
...
}
//请求数据成功
public void onRequestDataSuccess(Data successData){
fragmentList.get(currentPosition).setData(successData);
}
正常逻辑是没有问题的,但是当应用切换到后台,由于系统资源不足被杀掉的情况下,点击图标或任务列表返回app时,fragment显示空白,即setData无效。通过在创建fragment时和fragment的onCreateView方法打印日志发现,系统杀死app后重新创建时,onCreateView中打印的fragment不在遍历创建的fragmentList其中,即真正在viewpager中创建的fragment不在fragmentList之中
最后在FragmentPagerAdapter的instantiateItem方法中发现了问题
public Object instantiateItem(@NonNull ViewGroup container, int position) {
if (this.mCurTransaction == null) {
this.mCurTransaction = this.mFragmentManager.beginTransaction();
}
long itemId = this.getItemId(position);
String name = makeFragmentName(container.getId(), itemId);
Fragment fragment = this.mFragmentManager.findFragmentByTag(name);
//当activity重建时 此fragment不为null
if (fragment != null) {
this.mCurTransaction.attach(fragment);
} else {
fragment = this.getItem(position);
this.mCurTransaction.add(container.getId(), fragment, makeFragmentName(container.getId(), itemId));
}
if (fragment != this.mCurrentPrimaryItem) {
fragment.setMenuVisibility(false);
fragment.setUserVisibleHint(false);
}
return fragment;
}
makeFragmentName代码如下
private static String makeFragmentName(int viewId, long id) {
return "android:switcher:" + viewId + ":" + id;
}
当activity重建时会恢复已添加的fragments,当viewpager实例化当前fragment时并不都是通过getItem()返回的,还有以viewId+index为key值从已添加fragments中直接返回的,所以在activity重建时会重新遍历创建fragment添加到fragmentList中,而在viewpager中实际显示的fragment是从已添加的fragments中直接返回的fragment。
解决办法如下:
在activity中添加makeFragmentName方法
private String makeFragmentName(int viewId, long id) {
return "android:switcher:" + viewId + ":" + id;
}
在遍历创建fragment时加上判断,如果以makeFragmentName方法为key值的fragment已被添加,则直接返回,并添加到fragmentList中
List<Fragment> fragmentList = new ArrayList();
for(int i = 0;i< mData.size();i++){
Fragment fragmentByTag = getSupportFragmentManager().findFragmentByTag(makeFragmentName(R.id.view_pager, i));
if(fragmentByTag != null){
fragmentList.add(fragmentByTag);
}else{
fragmentList.add(MyFragment.newInstance(i));
}
}
还有一个需要注意的地方,fragment创建传值一定要通过setArguments方法传值,否则当fragment重建时,参数会丢失,希望大家注意
参考:
Android后台杀死系列
https://www.jianshu.com/p/00fef8872b68
https://stackoverflow.com/questions/47108494/how-to-restore-viewpager-from-savedinstancestate