初识ViewPager

引言

最近在工作中由于需要客制化系统的关系,接触到了很多ViewPager相关的UI,发现很多底层原生的界面也还是依然采用ViewPager+Fragment的布局方式,事实上这依然还是主流,微信5.0到6.0等还是采用这种布局方法,所以还是有必要总结下ViewPager的相关知识,增强下记忆和理解,毕竟伴随着经验的提升即使再去阅读同一本书也会有不一样的体会。

一、ViewPager

1、概述

ViewPager继承自ViewGroup,是左右两个屏幕平滑地切换的一个容器,容器里呈现的视图由对应的Adapter决定,和其他标准的AdapterView类似。简而言之就是我们通过Adapter把View放到ViewPager里,动动手指我们就可以实现左右滑动互相切换View了。另外ViewPager的更新不是直接由ViewPager本身去完成的,而是通过观察者去调用PagerAdapter的notifyDataSetChanged等相关方法去完成界面更新工作。

2、重要的成员

成员 说明
public ViewPager (Context context)
public ViewPager (Context context, AttributeSet attrs)
interface ViewPager.OnPageChangeListener 内部接口当切换不同Page时触发对应回调方法
void addView(View child, int index, ViewGroup.LayoutParams params) 添加View到ViewPager里
boolean dispatchKeyEvent(KeyEvent event) 派发键盘事件到下一个视图
ViewGroup.LayoutParams generateLayoutParams(AttributeSet attrs)
PagerAdapter getAdapter() 获取对应的适配器
int getCurrentItem() 返回当前page的index
int getOffscreenPageLimit() Returns the number of pages that will be retained to either side of the current page in the view hierarchy in an idle state.
int getPageMargin() 返回Page之间的外间距
boolean onTouchEvent(MotionEvent ev) 重写这个方法可以处理Touch事件
void removeView(View view) 注意不能在 draw(android.graphics.Canvas), onDraw(android.graphics.Canvas), dispatchDraw(android.graphics.Canvas)或其他相关方法调用
void setPageTransformer(boolean reverseDrawingOrder, ViewPager.PageTransformer transformer) 当the scroll position改变时触发,可以用于设置自定义的动画效果,即只要实现PageTransformer接口和其唯一的方法transformPage(View view, float position)
void setOnPageChangeListener(ViewPager.OnPageChangeListener listener) 设置Page切换事件监听
void setPageMarginDrawable(int resId)
void setAdapter(PagerAdapter adapter) 设置Adapter

二、重要的PagerAdapter

PagerAdapter其实是一个抽象类封装了一些接口、回调和功能方法,在实际开发过程中我们往往是使用它及其子类——FragmentPagerAdapterFragmentStatePagerAdapter的子类来填充到ViewPager中,通俗点来说,ViewPager只是个空皮囊,只有填充了PagerAdapter才能发挥其作用,正所谓巧妇也难为无米之炊,PagerAdapter就是这里的米,有了米才能各显神通做出各种各样的饭。

1111.png

1、实现PagerAdapter时必须要重写的四个成员方法

ViewPager不是直接与View关联的,而是通过一个key对象,通过这个key对象我们可以追踪并且唯一识别指定的page在PagerAdapter的位置。

  • Object instantiateItem(ViewGroup container, int position)——实例化item,PagerAdapter将选择将这个对象填充到在当前ViewPager里。

  • void destroyItem(ViewGroup container, int position, Object object)——将item从指定的位置移出容器

  • int getCount()——获取item的总数,一般都是获取集合的size

  • isViewFromObject(View,Object)——判断容器里的View是否与一个key值相关联,比如说例如Fragment+ViewPager的处理,每个页面都由它是对应的Fragment来呈现,但ViewPager并不是直接与View关联,而是关联一个key。我们实现这个方法也很简单谷歌推荐我们只需一句——return view == object;

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;  
    }  
}

2、PagerAdapter重要的成员方法

PagerAdapter的工作过程其实很简单,当ViewPager里的内容要改变的时候,就调用startUpdate方法来完成,随即也会多次调用instantiateItem或者destroyItem方法,finishUpdate最后被调用来完成更新工作。finishUpdate最终会返回需要添加到容器里的View和对应的 key对象(通过instantiateItem完成);而一些传递到destroyItem的则被移除。

成员方法 说明
void finishUpdate(ViewGroup container) 当完成page的更新工作时被调用
int getItemPosition(Object object) 但需要确定item位置是否改变时被调用,默认实现永远返回-1
void notifyDataSetChanged() 但需要去更新的时候我们可以去调用这个方法完成
void registerDataSetObserver(DataSetObserver observer) 注册观察者
void unregisterDataSetObserver(DataSetObserver observer) 取消注册
void restoreState(Parcelable state, ClassLoader loader)
Parcelable saveState()

3、FragmentPagerAdapter

FragmentPagerAdapter继承自PagerAdapter,主要的成员方法功能都是大同小异的,这个适配器主要就是用于快速实现Fragment在ViewPager里面进行滑动切换的,所以,如果我们想实现Fragment的左右滑动,优先选择ViewPager+FragmentPagerAdapter模式,因为FragmentPagerAdapter拥有自己的缓存策略,当配合ViewPager配合使用时,会缓存当前Fragment以及左边一个、右边一个(一共三个Fragment对象)假设有三个Fragment,那么在ViewPager初始化之后,3个fragment都会加载完成,中间的Fragment在整个生命周期里面只会加载一次,当最左边的Fragment处于显示状态,最右边的Fragment由于超出缓存范围,会被销毁,当再次滑到中间的Fragment的时候,最右边的Fragment会被再次初始化。所以比较适合用户制作较少页面切换的tab界面(最多3个),FragmentPagerAdapter会对我们浏览过Fragment进行缓存,保存这些界面的临时状态,这样当我们左右滑动的时候,界面切换更加的流畅。但是,这样也会增加程序占用的内存。那么3个以上的话,谷歌推荐移步到FragmentStatePagerAdapter

3.1、当我们使用FragmentPagerAdapter的时候,它的宿主ViewPager必须设置对应的id,同时必须实现两个方法和构造方法

  • getCount()——返回的是ViewPager页面的数量,即Fragement的数量。

  • getItem(int position)——返回的是要显示的fragment对象。

//实现一个FragmentPagerAdapter最少只需要实现getCount和getItem方法即可
FragmentPagerAdapter fragmentadapter = new FragmentPagerAdapter(  
             getSupportFragmentManager()) {  
         @Override  
         public int getCount() {  
             return fragments.size();  //getCount()返回的是ViewPager页面的数量,多少个Fragement对应多少个页面
         }  
         @Override  
         public Fragment getItem(int position) {  
             return fragments.get(position);//返回的是要显示的fragment对象。
         }  
      }  
   };  

4、FragmentStatePagerAdapter

当然除了FragmentPagerAdapter之外,还有一个类也是专门用于快速实现多个(3个以上)Fragment在ViewPager里面进行适配并滑动切换的,即FragmentStatePagerAdapterFragmentStatePagerAdapter也直接继承自PagerAdapter的,其工作方式和ListView十分相似的。当Fragment对用户不可见的时候,整个Fragment会被销毁并且保存Fragment的保存状态。基于这样的特性,FragmentStatePagerAdapter比FragmentPagerAdapter更适合用于很多界面之间的转换,而且消耗更少的内存资源。

3.1、当我们使用FragmentStatePagerAdapter的时候,它的宿主ViewPager必须设置对应的id,同时必须实现两个方法

  • getCount()——返回的是ViewPager页面的数量,即Fragement的数量。

  • getItem(int position)——返回的是要显示的fragment对象。

5、关于使用中发现,在删除或者修改数据的时候,PagerAdapter无法像BaseAdapter那样仅通过notifyDataSetChanged方法通知刷新View。

通常情况下调用notifyDataSetChanged方法会让ViewPager通过Adapter的getItemPosition方法查询一遍所有child view,如果所有child view位置均为POSITION_NONE,表示所有的child view都不存在,ViewPager会调用destroyItem方法销毁,并且重新生成,从而加大系统开销,并在一些复杂情况下导致逻辑问题。特别是对于只是希望更新child view内容的时候,但造成了完全不必要的开销,

  • 列表内容如果是针对于child view比较简单的情况(例如仅有TextView、ImageView等,没有ListView等展示数据的情况),可以重写PagerAdapter的方法getItemPosition
@Override    
public int getItemPosition(Object object) {    
    return POSITION_NONE;    
}
  • 但复杂的情况则需要根据自己的需求来实现notifyDataSetChanged的功能,比如,在仅需要对某个View内容进行更新时,在instantiateItem()时,用View.setTag方法加入标志,在需要更新信息时,通过findViewWithTag的方法找到对应的View进行局部更新。

6、FragmentPagerAdapter和FragmentStatePagerAdapter

FragmentPagerAdapter适合在较少Fragment滑动切换的界面使用的,划过的fragment会保存在内存中,尽管已经划过。而FragmentStatePagerAdapter和ListView有点类似,只会保存当前界面,以及下一个界面和上一个界面(如果有),最多保存3个,其他会被销毁掉。最后要注意的是FragmentStatePagerAdapter可能不经意间会造成内存未正常回收,严重导致内存溢出,比如图片资源没有释放,资源引用问题。(之前在网上看到过EditText由于保存焦点导致Fragment未被释放而导致OOM,设置edtText.saveEanble(false)就可以解决此问题)。最后PagerAdapter都只是数据集,数据集的改变只能发生在主线程里并且以必须调用notifyDataSetChanged() 来通知完成更新。

小结

这篇主要是简单描述了下ViewPager的主要功能以及PagerAdapter的角色和一些简略的原理,由于篇幅问题,下一篇再结合实例来总结下。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,752评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,100评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,244评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,099评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,210评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,307评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,346评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,133评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,546评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,849评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,019评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,702评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,331评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,030评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,260评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,871评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,898评论 2 351

推荐阅读更多精彩内容