FragmentPagerAdapter与FragmentStatePagerAdapter使用详解与区别

今天搞ViewPage滑动Fragment,一滑动再一回来就重建。。。找了半天,以为生命周期搞错了,后来发现用的Adaper是FragmentStatePagerAdapter。。。

正好整理一下这方面的知识点。大笑

FragmentPagerAdapter是android-support-v4支持包里面出现的一个新的适配器,继承自PagerAdapter,是专门用来给支持包中出现的ViewPager进行数据适配的。
        FragmentPagerAdapter,见名知意,这个适配器就是用来实现Fragment在ViewPager里面进行滑动切换的,因此,如果我们想实现Fragment的左右滑动,可以选择ViewPager和FragmentPagerAdapter实现。
    FragmentPagerAdapter拥有自己的缓存策略,当和ViewPager配合使用的时候,会缓存当前Fragment以及左边一个、右边一个,一共三个Fragment对象。
    假如有三个Fragment,那么在ViewPager初始化之后,3个fragment都会加载完成,中间的Fragment在整个生命周期里面只会加载一次,当最左边的Fragment处于显示状态,最右边的Fragment由于超出缓存范围,会被销毁,当再次滑到中间的Fragment的时候,最右边的Fragment会被再次初始化。
    在当前版本来说,最适合用来做固定的较少数量的场合,比如说一个有3个tab标签的fragment滑动界面。FragmentPagerAdapter会对我们浏览过Fragment进行缓存,保存这些界面的临时状态,这样当我们左右滑动的时候,界面切换更加的流畅。但是,这样也会增加程序占用的内存。如果应用场景是更多的Fragment,请使用FragmentStatePagerAdapter。
    当我们使用FragmentPagerAdapter的时候,它的宿主ViewPager必须有一个id。
    如果要使用FragmentPagerAdapter,我们需要实现2个方法,如下所示。

FragmentPagerAdapter adapter = new FragmentPagerAdapter(  
                getSupportFragmentManager()) {  
  
            @Override  
            public int getCount() {  
                return fragments.size();  
            }  
  
            @Override  
            public Fragment getItem(int position) {  
                return fragments.get(position);  
            }  
            }  
  
        };  
除了FragmentPagerAdapter之外,还有一个类也是专门实现ViewPager的Fragment的数据适配的,叫做FragmentStatePagerAdapter。
    FragmentStatePagerAdapter是PagerAdapter的子类,这个适配器对实现多个Fragment界面的滑动是非常有用的,它的工作方式和listview是非常相似的。 当Fragment对用户不可见的时候,整个Fragment会被销毁,只会保存Fragment的保存状态。基于这样的特性,FragmentStatePagerAdapter比FragmentPagerAdapter更适合用于很多界面之间的转换,而且消耗更少的内存资源。
     同样的,宿主VIewPager也必须有一个id。
     如果要使用FragmentStatePagerAdapter,我们需要实现2个方法,getCount()返回的是ViewPager页面的数量,getItem()返回的是要显示的fragent对象。使用方法和FragmentPagerAdapter完全一样。

平常使用的FragmentPagerAdapter和FragmentStatePagerAdapter来自android.support.v4.app包用来构建ViewPager。
FragmentPagerAdapter更多的用于少量界面的ViewPager,比如Tab。划过的fragment会保存在内存中,尽管已经划过。而FragmentStatePagerAdapter和ListView有点类似,会保存当前界面,以及下一个界面和上一个界面(如果有),最多保存3个,其他会被销毁掉。

要注意的是FragmentStatePagerAdapter可能不经意间会造成内存未正常回收,严重导致内存溢出,比如图片资源没有释放,资源引用问题。(之前碰到过EditTextt由于保存焦点导致Fragment未被释放,以至于内存溢出,设置editText.saveEanble(false)就可以解决此问题)。

下面上源码:

FragmentPagerAdapter创建View的过程

@Override
public Object instantiateItem(ViewGroup container, int position) {
if (mCurTransaction == null) {
mCurTransaction = mFragmentManager.beginTransaction();
}

final long itemId = getItemId(position); //获取id

// Do we already have this fragment?
String name = makeFragmentName(container.getId(), itemId);
//生成fragment Tag,格式”android:switcher:” + viewId + “:” + id;也可以通过这个来获取一个Fragment的实例
Fragment fragment = mFragmentManager.findFragmentByTag(name);
//下面判断是否有该Fragment,如果有就直接Attach到Activity里面去,没有的话就通过getItem方法获取新的实例
if (fragment != null) {
if (DEBUG) Log.v(TAG, “Attaching item #” + itemId + “: f=” + fragment);
mCurTransaction.attach(fragment);
} else {
fragment = getItem(position);
if (DEBUG) Log.v(TAG, “Adding item #” + itemId + “: f=” + fragment);
mCurTransaction.add(container.getId(), fragment,
makeFragmentName(container.getId(), itemId));
}
if (fragment != mCurrentPrimaryItem) {
FragmentCompat.setMenuVisibility(fragment, false);
FragmentCompat.setUserVisibleHint(fragment, false);
}
return fragment;
}
FragmentPagerAdapter移去View的过程
@Override
public void destroyItem(ViewGroup container, int position, Object object) {
if (mCurTransaction == null) {
mCurTransaction = mFragmentManager.beginTransaction();
}
if (DEBUG) Log.v(TAG, “Detaching item #” + getItemId(position) + “: f=” + object
+ ” v=” + ((Fragment)object).getView());
mCurTransaction.detach((Fragment)object); //把fragment从Activity中detach掉
}
由上面注释代码可知FragmentPagerAdapter创建时候把Fragment Attach到Activity,让Fragment不可见的时候Detach掉,这时候只要是创建的Fragment都不会被销毁掉,可以通过这个Tag找到–”android:switcher:” + viewId + “:” + id。(期中viewId是viewpager id,如R.id.viewpager,id是position)。
FragmentStatePagerAdapter创建View的过程
@Override
public Object instantiateItem(ViewGroup container, int position) {
// If we already have this item instantiated, there is nothing
// to do. This can happen when we are restoring the entire pager
// from its saved state, where the fragment manager has already
// taken care of restoring the fragments we previously had instantiated.
if (mFragments.size() > position) {
//mFragments是用来保存Fragment的ArrayList
Fragment f = mFragments.get(position);
if (f != null) {
return f;
}
}

if (mCurTransaction == null) {
mCurTransaction = mFragmentManager.beginTransaction();
}

Fragment fragment = getItem(position);
if (DEBUG) Log.v(TAG, “Adding item #” + position + “: f=” + fragment);
if (mSavedState.size() > position) {
Fragment.SavedState fss = mSavedState.get(position);
if (fss != null) {
fragment.setInitialSavedState(fss); //恢复Fragment的状态
}
}
while (mFragments.size() mFragments.add(null);
}
FragmentCompat.setMenuVisibility(fragment, false);
mFragments.set(position, fragment); //把新生成的fragment按照对应的position保存到mFragments里面去
mCurTransaction.add(container.getId(), fragment);//把fragment Attach到Activity

return fragment;
}
FragmentStatePagerAdapter销毁View的过程
public void destroyItem(ViewGroup container, int position, Object object) {
Fragment fragment = (Fragment)object;

if (mCurTransaction == null) {
mCurTransaction = mFragmentManager.beginTransaction();
}
if (DEBUG) Log.v(TAG, “Removing item #” + position + “: f=” + object
+ ” v=” + ((Fragment)object).getView());
while (mSavedState.size() mSavedState.add(null);
}
mSavedState.set(position, mFragmentManager.saveFragmentInstanceState(fragment)); //保存fragment的状态,如果有设置的话
mFragments.set(position, null);//清空mFragments列表中该位置的fragment

mCurTransaction.remove(fragment);//把fragment移除
}
由上面注释代码可知FragmentStatePagerAdapter创建时候把Fragment存到mFragments里面,并且Attach到Activity;销毁的时候从mFragments中移除,并且从Activity中移除。
关于如何获取FragmentStatePagerAdapter的实例引用,它不能通过”android:switcher:” + viewId + “:” + id来获取,可以使用instantiateItem(mViewPager,position)来获取,而FragmentPagerAdapter两种方法都可以。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值