Fragment操作方法和生命周期的关系

相信对Android开发比较熟悉的同学或多或少都用到过Fragment,Fragment附着在Activity上,有自己独特的生命

周期。FragmentTransaction提供了很多操作Fragment的方法,如add()、replace()、attach()等,调用这些方法会触发

Fragment不同的生命周期。调用了这些方法却不知道Fragment当于什么状态是一件危险的事情,因此,有必要

对Fragment的操作方法和生命周期的对应关系理一理。

       通过FragmentTransaction操作Fragment主要有以下几种方式:

add()

添加一个Fragment到Activity中

remove()

从Activity中移除一个Fragment,如果被移除的Fragment没有被添加到回退栈,这个Fragment实例将会被销毁。

replace()

使用另一个Fragment替换当前的,实际上是先调用remove()再调用add()

hide()

隐藏当前的Fragment,设置为不可见,但是并不会销毁

show()

显示之前隐藏的Fragment,设置为可见

detach()

将Fragment从Activity中分离,会销毁其View,但不会销毁Fragment的实例

attach()

将从Activity中分离的Fragment,重新关联到Activity,重新创建View

       总体看来,Fragment的操作方式主要可以分为两类:

显示:add() 、replace() 、show() 、attach()

隐藏:remove() 、hide() 、detach()

       下面通过例子来详细分析这几种方法的不同。

1、add方法

  1. public class MainActivity extends FragmentActivity {  
  2.   
  3.     FragmentA fragmentA = new FragmentA();  
  4.     FragmentB fragmentB = new FragmentB();  
  5.     private Button btn;  
  6.     @Override  
  7.     protected void onCreate(Bundle savedInstanceState) {  
  8.         super.onCreate(savedInstanceState);  
  9.         setContentView(R.layout.activity_main);  
  10.   
  11.         FragmentTransaction fragmentTransaction = getSupportFragmentManager().beginTransaction();  
  12.         fragmentTransaction.add(R.id.main_framelayout, fragmentA);  
  13.         fragmentTransaction.commit();  
  14.   
  15.         btn = (Button) findViewById(R.id.btn);  
  16.         btn.setOnClickListener(new View.OnClickListener() {  
  17.             @Override  
  18.             public void onClick(View v) {  
  19.                 FragmentTransaction fragmentTransaction = getSupportFragmentManager().beginTransaction();  
  20.                 fragmentTransaction.add(R.id.main_framelayout, fragmentB);  
  21.                 fragmentTransaction.commit();  
  22.             }  
  23.         });  
  24.     }  
  25. }  
public class MainActivity extends FragmentActivity {

    FragmentA fragmentA = new FragmentA();
    FragmentB fragmentB = new FragmentB();
    private Button btn;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        FragmentTransaction fragmentTransaction = getSupportFragmentManager().beginTransaction();
        fragmentTransaction.add(R.id.main_framelayout, fragmentA);
        fragmentTransaction.commit();

        btn = (Button) findViewById(R.id.btn);
        btn.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                FragmentTransaction fragmentTransaction = getSupportFragmentManager().beginTransaction();
                fragmentTransaction.add(R.id.main_framelayout, fragmentB);
                fragmentTransaction.commit();
            }
        });
    }
}
       我们还定义了两个Fragment,就是在各个生命周期中打印日志,观察生命周期的调用情况,代码就不在这里列出了。执行结果如下:


       两个 Fragment依次调用从onAttach到onResume的生命周期方法,FragmentB的添加并没有影响到FragmentA。这时虽然添加了两个Fragment,但是我们只能看到FragmentB,FragmentA被覆盖了。按返回键返回桌面,看打印日志。

2、remove方法

       只对上面的代码做一行修改,将

fragmentTransaction.add(R.id.main_framelayout, fragmentB);

       改为

fragmentTransaction.remove(fragmentA);

       即把FragmentA add到Activity之后,又将其remove,看一下发生了什么。


       可以看出通过remove方法,fragment的View被从页面上移除,同时Fragment在内存中的对象也被销毁了。这与我们之前给出的remove()方法的说明是一致的。

3、replace方法

       还是只改那一行代码,将

fragmentTransaction.add(R.id.main_framelayout, fragmentB);

       改为

fragmentTransaction.replace(R.id.main_framelayout, fragmentB);

       执行结果怎样呢?


       像我们前面说的,replace方法相当于先执行了remove再执行add。FragmentA先被从页面上移除并销毁了,然后FragmentB被创建并显示在了页面上。

4、detach和attach方法

       这一次我们要验证两个方法,先把Fragment从页面上detach,然后再将其attach回去,所以将之前的那一行代码改成了两行,

fragmentTransaction.detach(fragmentA);
fragmentTransaction.attach(fragmentA);

       执行结果如下:


       日志中显示detach的过程只执行了onPause()、onStop()和onDestroyView()三个方法,并没有执行onDestroy()和onDetach()方法,也就是说Fragment对象并没有销毁,只是从Activity中将View移除。而attach执行相反的过程。

5、show和hide方法

       这两个方法并不会触发任何Fragment的生命周期,只是将该Fragment设置为是否可见。

       了解了上面这些内容,下面通过一个例子来加深一下理解。

       很多APP的首页都是支持左右滑动的,它的实现就是通过ViewPager加一个适配器,对应于使用Fragment的情况,常用的有FragmentPagerAdapter和FragmentStatePagerAdapter两种。

       它们有什么区别呢?先来看看源码。首先是FragmentPagerAdapter

  1. @Override  
  2. public Object instantiateItem(ViewGroup container, int position) {  
  3.     if (mCurTransaction == null) {  
  4.         mCurTransaction = mFragmentManager.beginTransaction();  
  5.     }  
  6.   
  7.     final long itemId = getItemId(position);  
  8.   
  9.     // Do we already have this fragment?  
  10.     String name = makeFragmentName(container.getId(), itemId);  
  11.     Fragment fragment = mFragmentManager.findFragmentByTag(name);  
  12.     if (fragment != null) {  
  13.         if (DEBUG) Log.v(TAG, "Attaching item #" + itemId + ": f=" + fragment);  
  14.         mCurTransaction.attach(fragment);  
  15.     } else {  
  16.         fragment = getItem(position);  
  17.         if (DEBUG) Log.v(TAG, "Adding item #" + itemId + ": f=" + fragment);  
  18.         mCurTransaction.add(container.getId(), fragment,  
  19.                 makeFragmentName(container.getId(), itemId));  
  20.     }  
  21.     if (fragment != mCurrentPrimaryItem) {  
  22.         fragment.setMenuVisibility(false);  
  23.         fragment.setUserVisibleHint(false);  
  24.     }  
  25.   
  26.     return fragment;  
  27. }  
  28.   
  29. @Override  
  30. public void destroyItem(ViewGroup container, int position, Object object) {  
  31.     if (mCurTransaction == null) {  
  32.         mCurTransaction = mFragmentManager.beginTransaction();  
  33.     }  
  34.     if (DEBUG) Log.v(TAG, "Detaching item #" + getItemId(position) + ": f=" + object  
  35.             + " v=" + ((Fragment)object).getView());  
  36.     mCurTransaction.detach((Fragment)object);  
  37. }  
    @Override
    public Object instantiateItem(ViewGroup container, int position) {
        if (mCurTransaction == null) {
            mCurTransaction = mFragmentManager.beginTransaction();
        }

        final long itemId = getItemId(position);

        // Do we already have this fragment?
        String name = makeFragmentName(container.getId(), itemId);
        Fragment fragment = mFragmentManager.findFragmentByTag(name);
        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) {
            fragment.setMenuVisibility(false);
            fragment.setUserVisibleHint(false);
        }

        return fragment;
    }

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

        从上面代码发现,FragmentPagerAdapter在调用instantiateItem时,如果Fragment对象还不存在,就创建一个add到Activity上,如果已经存在就直接调用attach()方法,把Fragment对象attach到Activity上。FragmentPagerAdapter 会将所有生成的 Fragment 对象通过 FragmentManager 保存起来备用,以后需要该 Fragment 时,都会从 FragmentManager 读取,而不会再次调用 getItem() 方法。在destroyItem方法中执行的是detach()方法,那么这个对象并不会销毁,还可以通过Tag找到它。

  1. @Override  
  2. public Object instantiateItem(ViewGroup container, int position) {  
  3.     // If we already have this item instantiated, there is nothing  
  4.     // to do.  This can happen when we are restoring the entire pager  
  5.     // from its saved state, where the fragment manager has already  
  6.     // taken care of restoring the fragments we previously had instantiated.  
  7.     if (mFragments.size() > position) {  
  8.         Fragment f = mFragments.get(position);  
  9.         if (f != null) {  
  10.             return f;  
  11.         }  
  12.     }  
  13.   
  14.     if (mCurTransaction == null) {  
  15.         mCurTransaction = mFragmentManager.beginTransaction();  
  16.     }  
  17.   
  18.     Fragment fragment = getItem(position);  
  19.     if (DEBUG) Log.v(TAG, "Adding item #" + position + ": f=" + fragment);  
  20.     if (mSavedState.size() > position) {  
  21.         Fragment.SavedState fss = mSavedState.get(position);  
  22.         if (fss != null) {  
  23.             fragment.setInitialSavedState(fss);  
  24.         }  
  25.     }  
  26.     while (mFragments.size() <= position) {  
  27.         mFragments.add(null);  
  28.     }  
  29.     fragment.setMenuVisibility(false);  
  30.     fragment.setUserVisibleHint(false);  
  31.     mFragments.set(position, fragment);  
  32.     mCurTransaction.add(container.getId(), fragment);  
  33.   
  34.     return fragment;  
  35. }  
  36.   
  37. @Override  
  38. public void destroyItem(ViewGroup container, int position, Object object) {  
  39.     Fragment fragment = (Fragment)object;  
  40.   
  41.     if (mCurTransaction == null) {  
  42.         mCurTransaction = mFragmentManager.beginTransaction();  
  43.     }  
  44.     if (DEBUG) Log.v(TAG, "Removing item #" + position + ": f=" + object  
  45.             + " v=" + ((Fragment)object).getView());  
  46.     while (mSavedState.size() <= position) {  
  47.         mSavedState.add(null);  
  48.     }  
  49.     mSavedState.set(position, mFragmentManager.saveFragmentInstanceState(fragment));  
  50.     mFragments.set(position, null);  
  51.   
  52.     mCurTransaction.remove(fragment);  
  53. }  
    @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) {
            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);
            }
        }
        while (mFragments.size() <= position) {
            mFragments.add(null);
        }
        fragment.setMenuVisibility(false);
        fragment.setUserVisibleHint(false);
        mFragments.set(position, fragment);
        mCurTransaction.add(container.getId(), fragment);

        return fragment;
    }

    @Override
    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() <= position) {
            mSavedState.add(null);
        }
        mSavedState.set(position, mFragmentManager.saveFragmentInstanceState(fragment));
        mFragments.set(position, null);

        mCurTransaction.remove(fragment);
    }


       与FragmentPagerAdapter不同的是,FragmentStatePagerAdapter是通过add和remove方法添加Fragment的,Fragment的实例会被反复的销毁和重建。

       FragmentPagerAdapter更多的用于相对静态的、少量界面的ViewPager,划过的fragment会保存在内存中,如果加载的fragment较多会占用大量的内存。而FragmentStatePagerAdapter适用于数据动态性较大、页面比较多的情况,它并不会保存所有的fragment,默认情况下只会保存当前fragment以及上一个和下一个,其他会被销毁掉。

       这里提到的默认保存多少个页面,可以通过ViewPager的 setOffscreenPageLimit (int limit) 来设置,limit参数表示保持当前页面前面的limit个页面和后面limit个页面。如果这个值设置的很大,页面不会被反复销毁重建,性能比较好,但是会占用很多的内存。如果设置的很小则刚好相反。


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值