实例探究Android应用编写时Fragment的生命周期问题

实例探究Android应用编写时Fragment的生命周期问题,第1张

概述管理fragment的生命周期有些像管理activity的生命周期。Fragment可以生存在三种状态:

管理fragment的生命周期有些像管理activity的生命周期。Fragment可以生存在三种状态:

Resumed:

Fragment在一个运行中的activity中并且可见。

Paused:

另一个activity处于最顶层,但是fragment所在的activity并没有被完全覆盖(顶层的activity是半透明的或不占据整个屏幕)。

Stoped:

Fragment不可见。可能是它所在的activity处于stoped状态或是fragment被删除并添加到后退栈中了。此状态的fragment仍然存在于内存中。

同样类似于activity,你可以把fragment的状态保存在一个Bundle中,在activity被recreated时就需用到这个东西。你可以在onSaveInstanceState()方法中保存状态并在onCreate()或onCreateVIEw()或onActivityCreated()中恢复。

Fragment与Activity的生命周期中最大的不同就是存储到后退栈中的过程。Activity是在停止时自动被系统压入停止栈,并且这个栈是被系统管理的;而fragment是被压入activity所管理的一个后退栈,并且只有你在删除fragment后并明确调用addToBackStack()方法时才被压入。

然而,管理fragment的生命周期与管理activity的生命周期极其相似。你所需要去思考的是activity的生命周期如何影响fragment的生命周期。


下面这张Fragment生命周期图大家应该看得很多了:

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

使用add方法切换时:
载入Fragment1

Fragment1 onCreateFragment1 onCreateVIEwFragment1 onStartFragment1 onResume

用以下代码切到Fragment2:

FragmentTransaction ft = getFragmentManager().beginTransaction();ft.hIDe(Fragment1);ft.add(R.ID.simple_fragment,Fragment2);ft.setTransition(FragmentTransaction.TRANSIT_FRAGMENT_OPEN);ft.commit();

Fragment1不走任何生命周期,但会调onHIDdenChanged方法

Fragment2 onCreateFragment2 onCreateVIEwFragment2 onStartFragment2 onResume

回到Fragment1,Remove Fragment2:

FragmentTransaction ft = getFragmentManager().beginTransaction();ft.remove(Fragment2);ft.show(Fragment1);ft.setTransition(FragmentTransaction.TRANSIT_FRAGMENT_OPEN);ft.commit();

Fragment1还是不走任何生命周期,调onHIDdenChanged方法

Fragment2 onPauseFragment2 onStopFragment2 onDestoryVIEwFragment2 onDestory

用这种方法切换,Fragment在隐藏时并不会走onDestoryVIEw,所以显示时也不会走onCreateVIEw,所有VIEw都一直保存在内存中。
用replace方法:
载入Fragment1生命周期与上面相同:

Fragment1 onCreateFragment1 onCreateVIEwFragment1 onStartFragment1 onResume

切到Fragment2:

FragmentTransaction ft = getFragmentManager().beginTransaction();ft.replace(R.ID.simple_fragment,Fragment2);ft.setTransition(FragmentTransaction.TRANSIT_FRAGMENT_OPEN);ft.commit();

这次的Fragment1走生命周期了

Fragment1 onPauseFragment1 onStopFragment1 onDestoryVIEwFragment1 onDestoryFragment2 onCreateFragment2 onCreateVIEwFragment2 onStartFragment2 onResume

真实打印出来可能是Fragment1和Fragment2混在一起的,可以看到,Fragment1走了onDestory,被完全回收了!
再切回到Fragment1

FragmentTransaction ft = getFragmentManager().beginTransaction();ft.replace(R.ID.simple_fragment,Fragment1);ft.setTransition(FragmentTransaction.TRANSIT_FRAGMENT_OPEN);ft.commit();
Fragment1 onCreateFragment1 onCreateVIEwFragment1 onStartFragment1 onResumeFragment2 onPauseFragment2 onStopFragment2 onDestoryVIEwFragment2 onDestory

Fragment1因为已经被回收,又走onCreate,Fragment2被回收。

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

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

ft.addToBackStack(null);

在replace时加上这行,可以把原来的Fragment放入栈中,走onDestoryVIEw方法,但不会onDestory,返回时,直接onCreateVIEw,不再onCreate.
返回直接调用popBackStack()方法:

getFragmentManager().popBackStack();

总结

以上是内存溢出为你收集整理的实例探究Android应用编写时Fragment的生命周期问题全部内容,希望文章能够帮你解决实例探究Android应用编写时Fragment的生命周期问题所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/web/1141239.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-05-31
下一篇 2022-05-31

发表评论

登录后才能评论

评论列表(0条)

保存