PagerAdapter+ViewPager
可实现Fragment之间的切换
ViewPager常用方法:
- setAdapter(PagerAdapter adapter) 设置适配器
- setOffscreenPageLimit(int limit) 在空闲状态下设置应保留到视图层次结构中当前页面任一侧的页数。
- setCurrentItem(int item) 设置当前页
- setPageTransformer(boolean reverseDrawingOrder, PageTransformer transformer) 每当更改滚动位置时将为每个附加页面调用的内容。
- setPageMargin(int marginPixels) 设置页面之间的边距。
PagerAdapter
需重写的方法:
- int getCount()——获取item的总数,一般都是获取集合的size
- isViewFromObject(View,Object)——判断容器里的View是否与一个key值相关联,比如说例如Fragment+ViewPager的处理,每个页面都由它是对应的Fragment来呈现,但ViewPager并不是直接与View关联,而是关联一个key。返回值:如果对应的是同一个View,返回True,否则返回False。我们实现这个方法也很简单谷歌推荐我们只需一句——return view == object;。
可选择重写的方法:
- Object instantiateItem(ViewGroup container, int position)——实例化item,PagerAdapter将选择将这个对象填充到在当前ViewPager里。
- void destroyItem(ViewGroup container, int position, Object object)——将item从指定的位置移出容器
重要成员方法:
- void finishUpdate(ViewGroup container) 当完成page的更新工作时被调用
- int getItemPosition(Object object) 但需要确定item位置是否改变时被调用,默认实现永远返回-1
- void notifyDataSetChanged() 但需要去更新的时候我们可以去调用这个方法完成
- void registerDataSetObserver(DataSetObserver observer) 注册观察者
- void unregisterDataSetObserver(DataSetObserver observer) 取消注册
例子:
public class ViewPagerAdapter extends PagerAdapter {
private List<View> mList;
public ViewPagerAdapter(List<View> list) {
this.mList = list;
}
@Override
public int getCount() {
if (mList != null && mList.size() > 0) {
return mList.size();
} else {
return 0;
}
}
@Override
public boolean isViewFromObject(View view, Object object) {
return view == object;
}
@Override
public void destroyItem(ViewGroup container, int position, Object object) {
container.removeView((View) object);
}
@Override
public Object instantiateItem(ViewGroup container, int position) {
container.addView(mList.get(position));
return mList.get(position);
}
@Override
public int getItemPosition(Object object) {
return POSITION_NONE;
}
}
FragmentPagerAdapter
实现Fragment在ViewPager里面进行滑动切换
拥有自己的缓存策略,当配合ViewPager配合使用时,会缓存当前Fragment以及左边一个、右边一个(一共三个Fragment对象)。左右滑动的时候,界面切换更加的流畅。但是,这样也会增加程序占用的内存
如果Fragment
需重写的方法:
- getCount()——返回的是ViewPager页面的数量,即Fragement的数量。
- getItem(int position)——返回的是要显示的fragment对象。
注意:
- Fragment3个以上的话,谷歌推荐移步到FragmentStatePagerAdapter
- 使用FragmentPagerAdapter+ViewPager时,切换回上一个Fragment页面时(已经初始化完毕),不会回调任何生命周期方法以及onHiddenChanged(),只有setUserVisibleHint(boolean isVisibleToUser)会被回调,所以如果你想进行一些懒加载,需要在这里处理。setUserVisibleHint调用时机与其生命周期无关。
- 在给ViewPager绑定FragmentPagerAdapter时,new FragmentPagerAdapter(fragmentManager)的FragmentManager,一定要保证正确,如果ViewPager是Activity内的控件,则传递getSupportFragmentManager(),如果是Fragment的控件中,则应该传递getChildFragmentManager()。只要记住ViewPager内的Fragments是当前组件的子Fragment这个原则即可。
- 你不需要考虑在“内存重启”的情况下,去恢复的Fragments的问题,因为FragmentPagerAdapter已经帮我们处理啦。
public class MyFragmentPagerAdapter extends FragmentPagerAdapter {
private List<Fragment> mFragments;
private List<String> titleList; //标题列表数组
public MyFragmentPagerAdapter(FragmentManager fm,List<Fragment> fragments) {
super(fm);
mFragments=fragments;
}
public void setTitleList(List<String> titleList) {
this.titleList = titleList;
}
@Override
public Fragment getItem(int arg0) {
return mFragments.get(arg0);
}
@Override
public int getCount() {
return mFragments.size();
}
@Nullable
@Override
public CharSequence getPageTitle(int position) {
return titleList.get(position);
}
}
FragmentStatePagerAdapter
当Fragment对用户不可见的时候,整个Fragment会被销毁并且保存Fragment的保存状态。而在页面需要显示时,生成新的页面。基于这样的特性,FragmentStatePagerAdapter比FragmentPagerAdapter更适合用于很多界面之间的转换,而且消耗更少的内存资源。
(实际测试:Fragment不可见时不会被销毁,Adapter重写getItemPosition返回POSITION_NONE,也不会销毁。)
参考:
Android进阶——ViewPager详解之初识ViewPager(一)、