fragment之生命周期

来源:互联网 发布:22级战舰升级数据 编辑:程序博客网 时间:2024/06/07 08:47

Fragment必须是依存与Activity而存在的,因此Activity的生命周期会直接影响到Fragment的生命周期。官网这张图很好的说明了两者生命周期的关系:


可以看到Fragment比Activity多了几个额外的生命周期回调方法:
onAttach(Activity)
当Fragment与Activity发生关联时调用。
onCreateView(LayoutInflater, ViewGroup,Bundle)
创建该Fragment的视图
onActivityCreated(Bundle)
当Activity的onCreate方法返回时调用
onDestoryView()
与onCreateView想对应,当该Fragment的视图被移除时调用
onDetach()
与onAttach相对应,当Fragment与Activity关联被取消时调用

注意:除了onCreateView,其他的所有方法如果你重写了,必须调用父类对于该方法的实现


不同方法调用对应的生命周期调用:

但最近在写PageManager(管理页面跳转),发现切换页面时,之前的页面走完onDestoryView就直接onDestory了,回来又重新onCreate,如果用hide和show的方式,都不走生命周期,看了ApiDemo代码,发现原因,整理一下.
切换Fragment有两种方式,一种是add新的,并把旧的hide,另一种是replace.
旧的Fragment为Fragment1,新的是Fragment2,忽略非关键生命周期。

使用add方法切换时:
载入Fragment1
Fragment1 onCreate
Fragment1 onCreateView
Fragment1 onStart
Fragment1 onResume
用以下代码切到Fragment2:

  1. FragmentTransaction ft = getFragmentManager().beginTransaction();
  2. ft.hide(Fragment1);
  3. ft.add(R.id.simple_fragment, Fragment2);
  4. ft.setTransition(FragmentTransaction.TRANSIT_FRAGMENT_OPEN);
  5. ft.commit();

Fragment1不走任何生命周期,但会调onHiddenChanged方法
Fragment2 onCreate
Fragment2 onCreateView
Fragment2 onStart
Fragment2 onResume
回到Fragment1,Remove Fragment2:

  1. FragmentTransaction ft = getFragmentManager().beginTransaction();
  2. ft.remove(Fragment2);
  3. ft.show(Fragment1);
  4. ft.setTransition(FragmentTransaction.TRANSIT_FRAGMENT_OPEN);
  5. ft.commit();

Fragment1还是不走任何生命周期,调onHiddenChanged方法
Fragment2 onPause
Fragment2 onStop
Fragment2 onDestoryView
Fragment2 onDestory
用这种方法切换,Fragment在隐藏时并不会走onDestoryView,所以显示时也不会走onCreateView,所有View都一直保存在内存中。
用replace方法:
载入Fragment1生命周期与上面相同:
Fragment1 onCreate
Fragment1 onCreateView
Fragment1 onStart
Fragment1 onResume
切到Fragment2:

  1. FragmentTransaction ft = getFragmentManager().beginTransaction();
  2. ft.replace(R.id.simple_fragment, Fragment2);
  3. ft.setTransition(FragmentTransaction.TRANSIT_FRAGMENT_OPEN);
  4. ft.commit();

这次的Fragment1走生命周期了
Fragment1 onPause
Fragment1 onStop
Fragment1 onDestoryView
Fragment1 onDestory
Fragment2 onCreate
Fragment2 onCreateView
Fragment2 onStart
Fragment2 onResume
真实打印出来可能是Fragment1和Fragment2混在一起的,可以看到,Fragment1走了onDestory,被完全回收了!
再切回到Fragment1

  1. FragmentTransaction ft = getFragmentManager().beginTransaction();
  2. ft.replace(R.id.simple_fragment, Fragment1);
  3. ft.setTransition(FragmentTransaction.TRANSIT_FRAGMENT_OPEN);
  4. ft.commit();

Fragment1 onCreate
Fragment1 onCreateView
Fragment1 onStart
Fragment1 onResume
Fragment2 onPause
Fragment2 onStop
Fragment2 onDestoryView
Fragment2 onDestory
Fragment1因为已经被回收,又走onCreate,Fragment2被回收。

这两种方式显然都不满足我的需求,且与生命周期图不同。因为我需要在用户看见/看不见Fragment时register和unregister BroadcastReceiver之类的东西(onHiddenChanged也能实现,但第一次载入显示,以及销毁时不会走onHiddenChanged方法),也不希望用户回到上一个Fragment就重新创建整个Fragment,因为这样消耗资源。

看了ApiDemo,发现也是用replace方法,但是,我少了一行:

  1. ft.addToBackStack(null);

在replace时加上这行,可以把原来的Fragment放入栈中,走onDestoryView方法,但不会onDestory,返回时,直接onCreateView,不再onCreate,按回退键时,会回退到栈中的下一个fragment
如果要手动返回,直接调用popBackStack()方法:

  1. getFragmentManager().popBackStack();




0 0
原创粉丝点击