ViewPager详解

PagerAdapter+ViewPager

可实现Fragment之间的切换

ViewPager常用方法:

  • setAdapter(PagerAdapter adapter) 设置适配器
  • setOffscreenPageLimit(int limit) 在空闲状态下设置应保留到视图层次结构中当前页面任一侧的页数。
  • setCurrentItem(int item) 设置当前页
  • setPageTransformer(boolean reverseDrawingOrder, PageTransformer transformer) 每当更改滚动位置时将为每个附加页面调用的内容。
  • setPageMargin(int marginPixels) 设置页面之间的边距。

PagerAdapter

image.png

需重写的方法:

  • 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(一)

ViewPager 详解(一)---基本入门

FragmentPagerAdapter与FragmentStatePagerAdapter区别

Fragment全解析系列(二):正确的使用姿势

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容