服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - Android - Android的Fragment的生命周期各状态和回调函数使用

Android的Fragment的生命周期各状态和回调函数使用

2021-06-18 15:19Android开发网 Android

这篇文章主要介绍了Android的Fragments的生命周期各状态和回调函数使用,Fragments的生命周期与Activity息息相关,需要的朋友可以参考下

回调函数

就像activities一样,fragments也有它们自己的生命周期。理解fragments的生命周期,可以使你在它们被销毁的时候保存它们的实例,这样在它们重新被创建的时候,就能恢复它们之前的状态。

流程:
onattach()
作用:fragment已经关联到activity,

    这个是 回调函数

?
1
2
3
4
5
@override
public void onattach(activity activity) {
    super.onattach(activity);
    log.i("onattach_fragment");
}

    这个时候 activity已经传进来了, 获得activity的传递的值 就可以进行 与activity的通信里, 当然也可以使用getactivity(),前提是这个fragment已经和宿主的activity关联,并且没有脱离,他只调用一次。
oncreate()
系统创建fragment的时候回调他,在他里面实例化一些变量
这些个变量主要是:当你 暂停 停止的时候 你想保持的数据
如果我们要为fragment启动一个后台线程,可以考虑将代码放于此处。
参数是:bundle savedinstance, 用于保存 fragment 参数, fragement 也可以 重写 onsaveinstancestate(bundleoutstate) 方法, 保存fragement状态;
可以用于 文件保护
他只调用一次。

oncreateview()

    第一次使用的时候 fragment会在这上面画一个layout出来, 为了可以画控件 要返回一个 布局的view,也可以返回null。

    当系统用到fragment的时候 fragment就要返回他的view,越快越好 ,所以尽量在这里不要做耗时操作,比如从数据库加载大量数据显示listview,
    当然线程还是可以的。

    给当前的fragment绘制ui布局,可以使用线程更新ui,说白了就是加载fragment的布局的。
    这里一般都先判断是否为null。

?
1
2
3
4
5
6
7
if(text==null){
      bundle args=getarguments();
      text=args.getstring("text");
    }
    if (view == null) {
      view = inflater.inflate(r.layout.hello, null);
    }

这样进行各判断省得每次都要加载,减少资源消耗

onactivitycreated()

    当activity中的oncreate方法执行完后调用。   

    注意了:
    从这句官方的话可以看出:当执行onactivitycreated()的时候 activity的
    oncreate才刚完成。
    所以在onactivitycreated()调用之前 activity的oncreate可能还没有完成,
    所以不能再oncreateview()中进行 与activity有交互的ui操作,ui交互操作可以在onactivitycreated()里面进行。
    所以呢:这个方法主要是初始化那些你需要你的父activity或者fragment的ui已经被完
    整初始化才能初始化的元素。
    如果在oncreateview里面初始化空间 会慢很多,比如listview等。
onstart()

    和activity一致,启动fragement 启动时回调,,此时fragement可见。
onresume()

    和activity一致  在activity中运行是可见的。激活, fragement 进入前台, 可获取焦点时激活。
onpause()

    和activity一致  其他的activity获得焦点,这个仍然可见
    第一次调用的时候,指的是 用户 离开这个fragment(并不是被销毁)
    通常用于 用户的提交(可能用户离开后不会回来了)
onstop()

    和activity一致, fragment不可见的, 可能情况:activity被stopped了or fragment被移除但被,加入到回退栈中,一个stopped的fragment仍然是活着的如果长时间不用也会被移除。
ondestroyview()

    fragment中的布局被移除时调用。
    表示fragemnt销毁相关联的ui布局, 清除所有跟视图相关的资源。

    以前以为这里没什么用处其实 大有文章可做,
    相信大家都用过viewpager+fragment,由于viewpager的缓存机制,每次都会加载3
    页。
    例如:有四个 fragment 当滑动到第四页的时候 第一页执行ondestroyview(),但没有
    执行ondestroy。他依然和activity关联。当在滑动到第一页的时候又执行了
    oncreateview()。 生命周期可以自己试一下。
    那么问题来了。会出现重复加载view的局面,所以这么做(下面是先人的代码)

?
1
2
3
4
5
6
7
8
@override
  public void ondestroyview() {
    log.i("ondestroyview_fragment");
    if(view!=null){
            ((viewgroup)view.getparent()).removeview(view);
    }
    super.ondestroyview();
  }

ondestroy()

    销毁fragment对象, 跟activity类似了。
ondetach()

    fragment和activity解除关联的时候调用。 脱离activity。
可见fragment的销毁还是很优雅地,一个一个的来。

下面贴一下 activity和fragment同时运行时候的 生命周期

开始启动:

?
1
2
3
4
5
6
7
8
9
05-07 05:55:08.553: i/log(1990): oncreate
05-07 05:55:08.553: i/log(1990): onattach_fragment
05-07 05:55:08.553: i/log(1990): oncreate_fragment
05-07 05:55:08.553: i/log(1990): oncreateview_fragment
05-07 05:55:08.553: i/log(1990): onactivitycreated_fragment
05-07 05:55:08.553: i/log(1990): onstart
05-07 05:55:08.553: i/log(1990): onstart_fragment
05-07 05:55:08.553: i/log(1990): onresume
05-07 05:55:08.553: i/log(1990): onresume_fragment

按下home按键

?
1
2
3
4
05-07 05:55:28.725: i/log(1990): onpause_fragment
05-07 05:55:28.725: i/log(1990): onpause
05-07 05:55:29.221: i/log(1990): onstop_fragment
05-07 05:55:29.221: i/log(1990): onstop

再回到界面

?
1
2
3
4
5
05-07 05:55:49.441: i/log(1990): onrestart
05-07 05:55:49.441: i/log(1990): onstart
05-07 05:55:49.441: i/log(1990): onstart_fragment
05-07 05:55:49.441: i/log(1990): onresume
05-07 05:55:49.441: i/log(1990): onresume_fragment

销毁activity

?
1
2
3
4
5
6
7
8
05-07 05:59:02.293: i/log(1990): onpause_fragment
05-07 05:59:02.293: i/log(1990): onpause
05-07 05:59:02.757: i/log(1990): onstop_fragment
05-07 05:59:02.757: i/log(1990): onstop
05-07 05:59:02.757: i/log(1990): ondestroyview_fragment
05-07 05:59:02.757: i/log(1990): ondestroy_fragment
05-07 05:59:02.757: i/log(1990): ondetach_fragment
05-07 05:59:02.757: i/log(1990): ondestroy

可以看出 当现实fragment的时候都先执行activity方法,当销毁的时候都是现执行 fragment的方法,这样更好理解fragment是嵌套在activity中

下面一个综合性的例子测试了fragments的不同状态:
1.创建一个fragment的子类:fragment1.java。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
package net.horsttnann.fragments;
 
import android.app.activity;
import android.app.fragment;
import android.os.bundle;
import android.util.log;
import android.view.layoutinflater;
import android.view.view;
import android.view.viewgroup;
 
public class fragment1 extends fragment {
  @override
  public view oncreateview(layoutinflater inflater, viewgroup container,
      bundle savedinstancestate) {
 
    log.d("fragment 1", "oncreateview");
 
    // ---inflate the layout for this fragment---
    return inflater.inflate(r.layout.fragment1, container, false);
  }
 
  @override
  public void onattach(activity activity) {
    super.onattach(activity);
    log.d("fragment 1", "onattach");
  }
 
  @override
  public void oncreate(bundle savedinstancestate) {
    super.oncreate(savedinstancestate);
    log.d("fragment 1", "oncreate");
  }
 
  @override
  public void onactivitycreated(bundle savedinstancestate) {
    super.onactivitycreated(savedinstancestate);
    log.d("fragment 1", "onactivitycreated");
  }
 
  @override
  public void onstart() {
    super.onstart();
    log.d("fragment 1", "onstart");
  }
 
  @override
  public void onresume() {
    super.onresume();
    log.d("fragment 1", "onresume");
  }
 
  @override
  public void onpause() {
    super.onpause();
    log.d("fragment 1", "onpause");
  }
 
  @override
  public void onstop() {
    super.onstop();
    log.d("fragment 1", "onstop");
  }
 
  @override
  public void ondestroyview() {
    super.ondestroyview();
    log.d("fragment 1", "ondestroyview");
  }
 
  @override
  public void ondestroy() {
    super.ondestroy();
    log.d("fragment 1", "ondestroy");
  }
 
  @override
  public void ondetach() {
    super.ondetach();
    log.d("fragment 1", "ondetach");
  }
}

2.按ctrl+f11,将模拟器转换成“横屏模式”。
3.按f11调试。
4.当应用被加载的时候,logcat窗口中有如下显示。

?
1
2
3
4
5
6
03-27 12:23:32.255: d/fragment 1(704): onattach
03-27 12:23:32.255: d/fragment 1(704): oncreate
03-27 12:23:32.255: d/fragment 1(704): oncreateview
03-27 12:23:32.274: d/fragment 1(704): onactivitycreated
03-27 12:23:32.274: d/fragment 1(704): onstart
03-27 12:23:32.286: d/fragment 1(704): onresume

5.按home键,logcat窗口中有如下显示。

?
1
2
03-27 12:25:23.174: d/fragment 1(704): onpause
03-27 12:25:25.174: d/fragment 1(704): onstop

6.按home键不放,重新进入应用,logcat窗口中有如下显示。

?
1
2
03-27 12:26:21.505: d/fragment 1(704): onstart
03-27 12:26:21.505: d/fragment 1(704): onresume

7.按返回键,logcat窗口中有如下显示。

?
1
2
3
4
5
03-27 12:27:54.384: d/fragment 1(704): onpause
03-27 12:27:55.324: d/fragment 1(704): onstop
03-27 12:27:55.324: d/fragment 1(704): ondestroyview
03-27 12:27:55.324: d/fragment 1(704): ondestroy
03-27 12:27:55.324: d/fragment 1(704): ondetach

解析:
1.当一个fragment被创建的时候,它会经历以下状态.。
onattach()
oncreate()
oncreateview()
onactivitycreated()
2.当这个fragment对用户可见的时候,它会经历以下状态。
onstart()
onresume()
3.当这个fragment进入“后台模式”的时候,它会经历以下状态。
onpause()
onstop()
4.当这个fragment被销毁了(或者持有它的activity被销毁了),它会经历以下状态。
onpause()
onstop()
ondestroyview()
ondetach()
5.就像activitie一样,在以下的状态中,可以使用bundle对象保存一个fragment的对象。
oncreate()
oncreateview()
onactivitycreated()
6.fragments的大部分状态都和activitie很相似,但fragment有一些新的状态。
onattached() —— 当fragment和activity关联之后,调用这个方法。
oncreateview() —— 创建fragment中的视图的时候,调用这个方法。
onactivitycreated() —— 当activity的oncreate()方法被返回之后,调用这个方法。
ondestroyview() —— 当fragment中的视图被移除的时候,调用这个方法。
ondetach() —— 当fragment和activity分离的时候,调用这个方法。

延伸 · 阅读

精彩推荐