android ViewPager子頁面爲Fragment,app被殺死後重建引發的bug

項目中一個頁面中包含一個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

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章