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