android ViewPager子页面为Fragment,app被杀死后重建引发的bug

1,922 阅读2分钟

项目中一个页面中包含一个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后台杀死系列 www.jianshu.com/p/00fef8872…

stackoverflow.com/questions/4…