kniost

谁怕,一蓑烟雨任平生

0%

Android编程权威指南(第二版)学习笔记(十一)—— 第11章 使用 ViewPager

本章介绍了如何使用 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 类,在其中直接重写了两个关键方法,然后就可以使用了。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
// 引用 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(...)方法,而是使用下列方法:

1
2
3
4
5
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)方法的具体实现:

1
2
3
4
@Override
public boolean isViewFromObject(View view, Object object) {
return ((Fragment)object).getView() == view;
}

可以看到,每次需要使用ViewPager时,都要覆盖实现PagerAdapter的这些方法,这真是一种磨难。所幸我们有 FragmentPagerAdapterFragmentStatePagerAdapter 这么便利的类。


GitHub Page: kniost.github.io
简书:http://www.jianshu.com/u/723da691aa42