本章介绍了如何使用 ViewPager(准确地说,应该是使用了 FragmentStatePagerAdapter 的简单的 ViewPager)。
GitHub 地址:
完成第十一章
1. ViewPager 和 PagerAdapter
ViewPager 在某种程度上类似于 RecyclerView,它们都需要借助于 Adapter 来支持,ViewPager 需要的是 PagerAdapter。
ViewPager 与 PagerAdapter 之间的配合比 RecyclerView 与其 Adapter 之间复杂得多。但是对于本章而言,因为使用的是 PagerAdapter 的子类 FragmentStatePagerAdapter
,它能协助处理很多细节问题.
FragmentStatePagerAdapter
化繁为简,提供了两个有用的方法:getCount()
和 getItem (int)
。
调用 getCount()
方法顾名思义就是获取数据集的大小。调用 getItem(int)
方法,返回的是应该是和数据绑定的Fragment,一般来说会将其和数据集的位置相对应。
使用步骤:
- 布局文件,使用 ViewPager(因为只有支持包而没有内置,所以不像 fragment 需要选择)
- 在代码中声明 ViewPager 变量并引用
- 本书中使用的是匿名 FragmentStatePagerAdapter 类,在其中直接重写了两个关键方法,然后就可以使用了。
// 引用 ViewPager
mViewPager = (ViewPager) findViewById(R.id.activity_crime_pager_view_pager);
// 获取数据集
mCrimes = CrimeLab.get(this).getCrimes();
// 获取 FragmentManager
FragmentManager fragmentManager = getSupportFragmentManager();
// 使用匿名内部类来引用 FragmentStatePagerAdapter,构造方法的参数是 FragmentManager
mViewPager.setAdapter(new FragmentStatePagerAdapter(fragmentManager) {
@Override
public Fragment getItem(int position) {
Crime crime = mCrimes.get(position);
return CrimeFragment.newInstance(crime.getId());
}
@Override
public int getCount() {
return mCrimes.size();
}
});
// 设置完 Adapter 以后,就要选择当前的数据啦~这就是上一个 Activity 传递进来的数据,我就不予赘述了。
for (int i = 0; i < mCrimes.size(); i++) {
if (mCrimes.get(i).getId().equals(crimeId)) {
mViewPager.setCurrentItem(i);
break;
}
}
2. FragmentStatePagerAdapter 与 FragmentPagerAdapter
FragmentPagerAdapter
是另外一种可用的 PagerAdapter , 其用法与 FragmentStatePagerAdapter
基本一致。唯一的区别在于:卸载不再需要的 fragment 时,各自采用的处理方法有所不同。
FragmentStatePagerAdapter
会销毁不需要的 fragment。事务提交后,activity 的 FragmentManager
中的 fragment 会被彻底移除。FragmentStatePagerAdapter
类名中的“state”表明:在销毁 fragment 时,可在 onSaveInstanceState(Bundle)
方法中保存 fragment 的 Bundle 信息。用户切换回来时,保存的实例状态可用来恢复生成新的fragment。
相比之下,FragmentPagerAdapter
有不同的做法。对于不再需要的 fragment,FragmentPagerAdapter
会选择调用事务的 detach(Fragment) 方法来处理它,而非 remove(Fragment) 方 法。也就是说,FragmentPagerAdapter
只是销毁了 fragment 的视图,fragment 实例还保留在 FragmentManager
中。因此,FragmentPagerAdapter
创建的 fragment 永远不会被销毁。
选择哪种adapter取决于应用的要求。通常来说,使用 FragmentStatePagerAdapter
更节省内存。CriminalIntent 应用需显示大量crime记录,每份记录最终还会包含图片。在内存中保存所有信息显然不合适,因此我们选择使用 FragmentStatePagerAdapter
。
另一方面,如果用户界面只需要少量固定的fragment,则 FragmentPagerAdapter
是个安全、 合适的选择。
最常见的例子为分页显示用户界面。例如,某些应用的明细视图所含内容较多,通 常需分两页显示。这时就可以将这些明细信息分拆开来,以多页面的形式展现。显然,为用户界面添加支持滑动切换的 ViewPager,能增强应用的触摸体验。此外,将 fragment 保存在内存中,更易于管理控制层的代码。对于这种类型的用户界面,每个 activity 通常只有两三个 fragment,基本不用担心有内存不足的风险。
3. 深入学习:ViewPager 的工作原理
什么时候需要自己实现PagerAdapter接口呢?需要ViewPager托管非 fragment 视图时,就需要实现原生 PagerAdapter 接口。
PagerAdapter 要比 RecyclerView 的 Adapter复杂得多,因为它要处理更多的视图管理工作。
PagerAdapter 不使用可返回视图的onBindViewHolder(...)
方法,而是使用下列方法:
public Object instantiateItem(ViewGroup container, int position)
public void destroyItem(ViewGroup container, int position, Object object)
public abstract boolean isViewFromObject(View view, Object object)
-
PagerAdapter.instantiateItem(ViewGroup, int)
方法告诉 PagerAdapter 创建指定位置的列表项视图,然后将其添加给 ViewGroup 视图容器,而destroyItem(ViewGroup, int, Object)
方法则告诉 PagerAdapter 销毁已建视图。(注意,instantiateItem(ViewGroup, int)
方法并不要求立即创建视图。因此,PagerAdapter 可自行决定何时创建视图。) -
视图创建完成后,ViewPager 会在某个时间点注意到它。为确定该视图所属的对象,ViewPager 会调用
isViewFromObject(View, Object)
方法。这 里 , Object 参数是instantiateItem(ViewGroup,int)
方法返回的对象。因此,假设 ViewPager 调用instantiateItem(ViewGroup, 5)
方法返回一个 A 对象,那么只要传入的 View 参数是第5个对象的视图,isViewFromObject(View, A)
方法就应返回true值,否则返回false值。
对 ViewPager 来说,这是一个复杂的过程,但对于PagerAdapter来说,这算不上什么。因为PagerAdapter只要能够创建、销毁视图以及识别视图来自哪个对象即可。这样的要求显然很宽松,因而PagerAdapter 能够比较自由地通过 instantiateItem(ViewGroup, int)
方法创建并添加新的fragment ,然后返回可以跟踪管理的 Object(fragment) 。
以下为isViewFromObject (View, Object)方法的具体实现:
@Override
public boolean isViewFromObject(View view, Object object) {
return ((Fragment)object).getView() == view;
}
可以看到,每次需要使用ViewPager时,都要覆盖实现PagerAdapter的这些方法,这真是一种磨难。所幸我们有 FragmentPagerAdapter
和 FragmentStatePagerAdapter
这么便利的类。
GitHub Page: kniost.github.io
简书:http://www.jianshu.com/u/723da691aa42
网友评论