通过ViewPager和它的Adapter如何工作来解析这二者的不同之处。
android.support.v4.app.FragmentPagerAdapter
继承自android.support.v4.view.PagerAdapter,每页都是一个Fragment,并且所有的Fragment实倒都保存在FragmentManager中。因此它适用于少量且固定的Fragment,比如一组用于分页显示的tab。除了当Fragment不可见时,它的视图层可能被销毁外,每页的Fragment都会被保存在内存中。(翻译自源码注释)android.support.v4.app.FragmentStatePagerAdapter
继承自android.support.v4.view.PagerAdapter,每页都是一个Fragment,当Fragment不可见时,整个Fragment都会被销毁,只有Fragment的 saved state会被保存下来,它适用于许多tab的情况,类似于ListView。(翻译自源码注释)
下面的代码是ViewPager与FragmentPagerAdapter组合使用的典型用法
package com.example.viewpager;
import android.os.Bundle;
import android.support.design.widget.TabLayout;
import android.support.v4.app.Fragment;
import android.support.v4.app.FragmentPagerAdapter;
import android.support.v4.view.ViewPager;
import android.support.v7.app.AppCompatActivity;
import butterknife.BindView;
import butterknife.ButterKnife;
public class MainActivity extends AppCompatActivity {
@BindView(R.id.tab_layout)
TabLayout mTabLayout;
@BindView(R.id.view_pager)
ViewPager mViewPager;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
ButterKnife.bind(this);
mViewPager.setAdapter(mFragmentPagerAdapter);
}
private FragmentPagerAdapter mFragmentPagerAdapter = new FragmentPagerAdapter(getSupportFragmentManager()) {
@Override
public Fragment getItem(int position) {
return new MyFragment();
}
@Override
public int getCount() {
return 2;
}
};
}
通过以下二张图很直观地比较FragmentPagerAdapter和FragmentStatePagerAdapter的区别
FragmentPagerAdapter的Fragment管理:
FragmentStatePagerAdapter的Fragment管理:
设置FragmentPagerAdapter
下面通过打印log进行分析
10-09 11:12:00.655 31912-31912/? I/MyFragment0: getItem: //刚开始在第1页
10-09 11:12:00.670 31912-31912/? I/MyFragment1: getItem:
10-09 11:12:00.673 31912-31912/? I/MyFragment: onAttach: fragment #1 //Fragment1
10-09 11:12:00.673 31912-31912/? I/MyFragment: onCreate: fragment #1 // Fragment1
10-09 11:12:00.673 31912-31912/? I/MyFragment: onAttach: fragment #2 //Fragment2
10-09 11:12:00.673 31912-31912/? I/MyFragment: onCreate: fragment #2 //Fragment2
10-09 11:12:00.678 31912-31912/? I/MyFragment: onCreateView: fragment #1 //Fragment1
10-09 11:12:00.682 31912-31912/? I/MyFragment: onViewCreated: fragment #1 //Fragment1
10-09 11:12:00.685 31912-31912/? I/MyFragment: onCreateView: fragment #2 //Fragment2
10-09 11:12:00.686 31912-31912/? I/MyFragment: onViewCreated: fragment #2 //Fragment2
从第1页滑到第2页之后
10-09 11:20:25.353 31912-31912/? I/MyFragment: onAttach: fragment #3 //Fragment3
10-09 11:20:25.353 31912-31912/? I/MyFragment: onCreate: fragment #3 //Fragment3
10-09 11:20:25.372 31912-31912/? I/MyFragment: onCreateView: fragment #3 //Fragment3
10-09 11:20:25.379 31912-31912/? I/MyFragment: onViewCreated: fragment #3 //Fragment3
从第2页滑到第3页之后
10-09 11:22:08.255 31912-31912/? I/MyFragment0: destroyItem: //Fragment0
10-09 11:22:08.256 31912-31912/? I/MyFragment: onDestroyView: fragment #1 //Fragment0
FragmentPagerAdapter和FragmentStatePagerAdapter对待下一页的方式是一样的,但是对于上上页的处理方式却不一样。以下是使用FragmentStatePagerAdapter打印出的Log。
设置FragmentStatePagerAdapter
切到第1页时
10-09 11:30:16.045 31912-31912/? I/MyFragment: onDetach: fragment #3
10-09 11:30:28.494 31912-31912/? I/MyFragment0: getItem:
10-09 11:30:28.494 31912-31912/? I/MyFragment1: getItem:
10-09 11:30:28.495 31912-31912/? I/MyFragment: onAttach: fragment #1
10-09 11:30:28.495 31912-31912/? I/MyFragment: onCreate: fragment #1
10-09 11:30:28.495 31912-31912/? I/MyFragment: onAttach: fragment #2
10-09 11:30:28.495 31912-31912/? I/MyFragment: onCreate: fragment #2
10-09 11:30:28.498 31912-31912/? I/MyFragment: onCreateView: fragment #1
10-09 11:30:28.499 31912-31912/? I/MyFragment: onViewCreated: fragment #1
10-09 11:30:28.500 31912-31912/? I/MyFragment: onCreateView: fragment #2
10-09 11:30:28.501 31912-31912/? I/MyFragment: onViewCreated: fragment #2
从第1页切到第2页时
10-09 11:32:00.712 31912-31912/? I/MyFragment: onAttach: fragment #3
10-09 11:32:00.712 31912-31912/? I/MyFragment: onCreate: fragment #3
10-09 11:32:00.714 31912-31912/? I/MyFragment: onCreateView: fragment #3
10-09 11:32:00.715 31912-31912/? I/MyFragment: onViewCreated: fragment #3
从第2页切到第3页时
-09 11:42:01.973 18087-18087/com.example.viewpager I/MyFragment: onAttach: fragment #3
10-09 11:42:01.973 18087-18087/com.example.viewpager I/MyFragment: onCreate: fragment #3
10-09 11:42:01.977 18087-18087/com.example.viewpager I/MyFragment: onCreateView: fragment #3
10-09 11:42:01.978 18087-18087/com.example.viewpager I/MyFragment: onViewCreated: fragment #3
10-09 11:42:03.922 18087-18087/com.example.viewpager I/MyFragment0: destroyItem:
10-09 11:42:03.923 18087-18087/com.example.viewpager I/MyFragment: onDestroyView: fragment #1
10-09 11:42:03.923 18087-18087/com.example.viewpager I/MyFragment: onDestroy: fragment #1
10-09 11:42:03.924 18087-18087/com.example.viewpager I/MyFragment: onDetach: fragment #1
Fragment getItem(int position)
/**
* Return the Fragment associated with a specified position.
*/
public abstract Fragment getItem(int position);
当adapter需要指定位置的Fragment时,而这个Fragment又不存在,那么getItem方法就会被调到,返回一个Fragment实例。需要注意的是,getItem方法是创建一个Fragment实例,而不是返回一个已经存在的Fragment实例。对于FragmentPagerAdapter,当每页的Fragment被创建后,这个函数就不会被调到了。对于FragmentStatePagerAdapter,由于Fragment会被销毁,所以它仍会被调到。 由于我们必须在getItem中实例化一个Fragment,所以当getItem()被调用后,Fragment相应的生命周期函数也就被调到了:
FragmentPagerAdapter和FragmentStatePagerAdapter对比总结
二者使用方法基本相同,唯一的区别就在卸载不再需要的fragment时,采用的处理方式不同:
- 使用FragmentStatePagerAdapter会销毁掉不需要的fragment。事务提交后,可将fragment从activity的FragmentManager中彻底移除。类名中的“state”表明:在销毁fragment时,它会将其onSaveInstanceState(Bundle) 方法中的Bundle信息保存下来。用户切换回原来的页面后,保存的实例状态可用于恢复生成新的fragment.
- FragmentPagerAdapter的做法大不相同。对于不再需要的fragment,FragmentPagerAdapter则选择调用事务的detach(Fragment) 方法,而非remove(Fragment)方法来处理它。也就是说,FragmentPagerAdapter只是销毁了fragment的视图,但仍将fragment实例保留在FragmentManager中。因此, FragmentPagerAdapter创建的fragment永远不会被销毁。
(摘自《Android权威编程指南11.1.4》)