摘自:安卓APP_ Fragment(3)—— Fragment的生命周期
作者:丶PURSUING
发布时间: 2021-04-16 22:32:12
网址:https://blog.csdn.net/weixin_44742824/article/details/115768202

目录

  • Fragment生命周期函数一览
  • 编程感受Fragment生命周期
    • 演示所用的代码
    • 各个事件对应的生命周期函数调用情况
      • (1)打开界面:触发生命周期
      • (2)点击home键,返回桌面
      • (3)重新打开fragment界面
      • (4)返回按键,返回到桌面
    • 前4种情况小结:
    • 重点理解:栈管理下生命周期函数调用情况
      • (1)打开界面,即创建过程与上面一样
      • (2)替换fragment
      • (3)按下返回后,BlankFragment1出栈
      • (4)BlankFragment2下返回桌面与重回界面
      • (5)关闭应用,即删除了Activity
  • Fragment生命周期注意事项再强调

Fragment生命周期函数一览

Activity和Fragment的生命周期非常类似,Fragment要更加细分一些,如下图:


由上图可知,所有的生命周期函数必须在 绑定(onAttach)解绑(onDetach) 两者之间执行。

Fragment的生命周期非常重要,在项目中Fragment生命周期的滥用,会导致后台收集到很多的异常,而异常的根本原因是对其生命周期没有按照规则执行,例如:在fragment中从Activity获取的变量为null。

编程感受Fragment生命周期

生命周期难点所在:生命周期的调度原则。

这些生命周期函数并不是每次fragment变化的时候都会全部调用,而是只会调用其中某几个,这就需要我们知道在各种情况下的调用情况。

根据调用情况,我们才清除解析bandle,解析xmlUI复位清零等等应该放在哪个生命周期函数中。

后面例子中有体现:

onCreate()要对从activity传过来的bundle进行解析等
onCreateView()要解析xml,设置button等

演示所用的代码

代码结构一览:

MainActivity.java

在这里创建了两个按钮的点击事件,点击后创建或者切换不同的fragment。

public class MainActivity extends AppCompatActivity implements View.OnClickListener {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);Button button1 = findViewById(R.id.btn_1);button1.setOnClickListener(this);Button button2 = findViewById(R.id.btn_2);button2.setOnClickListener(this);}@Overridepublic void onClick(View v) {switch (v.getId()) {case R.id.btn_1:Bundle bundle = new Bundle();//将Activity中的信息放进bundlebundle.putString("message", "my name is zhua");//需要实例化一个BlankFragment1对象bfBlankFragment1 bf = new BlankFragment1();//数据传入bf中bf.setArguments(bundle);//动态切换fragmentreplaceFragment(bf);break;case R.id.btn_2:replaceFragment(new BlankFragment2());break;}}//启动fragment生命周期private void replaceFragment(Fragment fragment) {FragmentManager fragmentManager = getSupportFragmentManager();FragmentTransaction transaction = fragmentManager.beginTransaction();transaction.replace(R.id.fm, fragment);transaction.addToBackStack(null);transaction.commit();}
}
  • 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

activity_main.xml

重点看FrameLayout布局,本例所用的两个fragment切换显示就在这个布局上。

<?xml version="1.0" encoding="utf-8"?><LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical"><Buttonandroid:layout_width="match_parent"android:layout_height="wrap_content"android:id="@+id/btn_1"android:text="change_1"/><Buttonandroid:layout_width="match_parent"android:layout_height="wrap_content"android:id="@+id/btn_2"android:text="change_2"/><!--    除去按钮,剩余的空间都是FrameLayout--><FrameLayoutandroid:layout_width="match_parent"android:layout_height="match_parent"android:id="@+id/fm"android:background="@color/teal_200"/></LinearLayout>
  • 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

BlankFragment1.java

这是本篇章的重点,把所有的fragment生命周期函数按照 一、Fragment生命周期函数一览 图片顺序编写。在每个生命周期函数中,都打印有调试信息,方便追踪在不同的执行状况下各个函数的调用情况。

public class BlankFragment1 extends Fragment {private String TAG = "zhua";//因为onCreateView函数可能会被调用多次,为了防止被解析多次而浪费资源,rootview要定义为全局private View rootview;@Overridepublic void onAttach(@NonNull Context context) {super.onAttach(context);Log.e(TAG, "onAttach: ");}@Overridepublic void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);//这个getArguments返回的是在MainActivity中传入的bundleBundle bundle = this.getArguments();//获取bundle里面保存的内容String string = bundle.getString("message");//在fragment中打印从Activity中传递来的信息Log.e(TAG, "onCreate: " );}@Overridepublic View onCreateView(LayoutInflater inflater, ViewGroup container,Bundle savedInstanceState) {Log.e(TAG, "onCreateView: ");//必须先进行判断,如果没有解析过再进行解析,防止被多次调用if(rootview == null){// 用inflater解析器进行xml解析        要渲染的布局     xml副本,直接containerrootview = inflater.inflate(R.layout.fragment_blank1,container,false);}/*onCreateView中可以对Button进行绑定(如果需要使用Button)Button button1 = rootview.findViewById(R.id.btn_1);button1.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {}});*/return rootview;}@Overridepublic void onActivityCreated(@Nullable Bundle savedInstanceState) {super.onActivityCreated(savedInstanceState);Log.e(TAG, "onActivityCreated: " );}@Overridepublic void onStart() {super.onStart();Log.e(TAG, "onStart: " );}@Overridepublic void onResume() {super.onResume();Log.e(TAG, "onResume: " );}@Overridepublic void onPause() {super.onPause();Log.e(TAG, "onPause: " );}@Overridepublic void onStop() {super.onStop();Log.e(TAG, "onStop: " );}@Overridepublic void onDestroyView() {super.onDestroyView();Log.e(TAG, "onDestroyView: " );}@Overridepublic void onDestroy() {super.onDestroy();Log.e(TAG, "onDestroy: " );}@Overridepublic void onDetach() {super.onDetach();Log.e(TAG, "onDetach: " );}
}
  • 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
  • 82
  • 83
  • 84
  • 85
  • 86
  • 87
  • 88
  • 89
  • 90
  • 91
  • 92
  • 93
  • 94
  • 95

BlankFragment2.java

public class BlankFragment2 extends Fragment {private View rootview;@Overridepublic void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);}@Overridepublic View onCreateView(LayoutInflater inflater, ViewGroup container,Bundle savedInstanceState) {//必须先进行判断,如果没有解析过再进行解析,防止被多次调用if(rootview == null){// 用inflater解析器进行xml解析        要渲染的布局     xml副本,直接containerrootview = inflater.inflate(R.layout.fragment_blank1,container,false);}return rootview;}
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22

fragment_blank1.xml

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"tools:context=".BlankFragment1"><!-- TODO: Update blank fragment layout --><TextViewandroid:layout_width="match_parent"android:layout_height="match_parent"android:background="#ffff00"android:textSize="40dp"android:text="@string/hello_blank_fragment" /></FrameLayout>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16

fragment_blank2.xml

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"tools:context=".BlankFragment2"><!-- TODO: Update blank fragment layout --><TextViewandroid:layout_width="match_parent"android:layout_height="match_parent"android:text="@string/hello_blank_fragment" /></FrameLayout>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14

各个事件对应的生命周期函数调用情况

(1)打开界面:触发生命周期

点击按钮,创建fragment,调用了如下六个函数:

即为

(2)点击home键,返回桌面

可以看到调用的是暂停和停止

如果要将UI进行复位,进行状态清0,那么就要放在onPause里面进行

(3)重新打开fragment界面

调用了开始与重启

(4)返回按键,返回到桌面


因为这个返回已经导致了Activity生命周期的结束,所以fragment暂停 > 停止 > 销毁布局 > 销毁fragment > 解除绑定

前4种情况小结:

生命周期的创建与销毁过程:是完全逆向的,一 一对应的过程。

重点理解:栈管理下生命周期函数调用情况

如果fragment是放进栈里面管理(相当于添加了一个保险箱),而不是直接销毁,那fragment生命周期又会有什么不同呢?

(1)打开界面,即创建过程与上面一样

点击的是CHANGE_1,此时界面是BlankFragment1

(2)替换fragment

点击第二个按钮切换fragment,此时界面是BlankFragment12

当替换的时候,就需要销毁view(onDestroyView,注意哦是删除了布局,而不是fragment),因为当前所展示的Activity里面的fragement已经变了。

(3)按下返回后,BlankFragment1出栈

按一下返回,切换回来,此时界面是BlankFragment1
可以发现(2)(3)非常像是逆着过来的过程。

(4)BlankFragment2下返回桌面与重回界面

在界面为BlankFragment2(即下图界面)执行1 2 3 操作

注意,打印信息仅仅在BlankFragment1中有设置

(5)关闭应用,即删除了Activity

可以看到fragment生命周期完全结束,销毁fragment并解绑。

Fragment生命周期注意事项再强调

(1)将来开发者会围绕fragment生命周期花费很多时间来解决问题,比如fragement切换白屏,显示异常等。

(2)fragement的使用一定是需要在生命周期函数onAttach与onDetach之间。

(3)fragement的使用一定要严格遵守生命周期的规则,在正确的地方写恰当的代码。

安卓APP_ Fragment(3)—— Fragment的生命周期相关推荐

  1. 3.1 Fragment理论知识及其生命周期,看这一片就够了

    点此进入:从零快速构建APP系列目录导图 点此进入:UI编程系列目录导图 点此进入:四大组件系列目录导图 点此进入:数据网络和线程系列目录导图 一.Fragment 是什么? Fragment 是一种 ...

  2. 史上最全Fragment介绍,包括fragment的定义,生命周期,用法

    一.为什么要使用Fragment 1.当我们需要动态的多界面切换的时候,就需要将UI元素和Activity融合成一个模块.在2.3中我们一般通过各种Activity中进行跳转来实现多界面的跳转和单个界 ...

  3. Activity、Fragment和Surface的生命周期

    这里首先推荐大家一本Android的学习书籍:<第一行代码>,这本书还是写得非常好的.好了,进入正题,我们先来了解一下Activity.Fragment的的生命周期中经历的几个阶段. Ac ...

  4. 安卓学习笔记06:Activity生命周期与启动模式

    文章目录 零.学习目标 一.Activity生命周期 1.了解Activity生命周期 2.Activity生命周期简化图 (1)Activity存在与否 (2)Activity可见与否 (3)Act ...

  5. 【词汇详解】“生命周期”之为什么线程(或者安卓的activity等)要有生命周期

    突然想到这个问题,,为什么线程或者安卓的activity等等,都要有个生命周期? 原因:在创建和销毁的中间有不同的状态,不同状态下可以干不同的事(解锁新功能). 线程,就绪状态>阻塞状态> ...

  6. 安卓服务service全解,生命周期,前台服务、后台服务,启动注销、绑定解绑,注册

    全栈工程师开发手册 (作者:栾鹏) python教程全解 定义服务(服务的生命周期) 调用context.startService()时依次执行 ->onCreate()- >onStar ...

  7. Activity与Fragment的生命周期详解

    在安卓中Activity与Fragment是非常相似的两个类,它们各自都拥有自己的生命周期,且都可以用来显示布局文件中的视图.其中Activity是通过setContenView()显示视图,而Fra ...

  8. Android中Fragment生命周期和基本用法

    1.基本概念 1. Fragment是什么? Fragment是可以让你的app纵享丝滑的设计,如果你的app想在现在基础上性能大幅度提高,并且占用内存降低,同样的界面Activity占用内存比Fra ...

  9. 友盟页面统计 - 关于Viewpager中的Fragment的生命周期

    Activity和Fragment各自理论上的生命周期 Activity的生命周期是较为经典也最清晰的,在此不表: Fragment从出现到广泛运用也有一段时间了,其标准生命周期也仅比Activity ...

  10. Lifecycle Activity和Fragment生命周期感知组件 LifecycleObserver MD

    Markdown版本笔记 我的GitHub首页 我的博客 我的微信 我的邮箱 MyAndroidBlogs baiqiantao baiqiantao bqt20094 baiqiantao@sina ...

最新文章

  1. 懒 人 世 界 的 生 存 法 则
  2. 费曼:所有的科学知识都是不确定的
  3. 高斯赛德尔迭代c语言_逐次超松弛SOR迭代概述
  4. 如何用web3.js在以太坊区块链上保存数据?
  5. Fault Tolerance 要求、限制和许可
  6. 微信小程序开发——以简单易懂的浏览器页面栈理解小程序的页面路由
  7. 油猴+百度网盘+加速
  8. 三阶魔方还原 - 只需7步6个公式
  9. 2021年9月国产数据库排行榜:达梦奋起直追紧逼OceanBase,openGauss反超PolarDB再升一位...
  10. 【樂理】鋼琴琴鍵的頻率
  11. 数据结构与算法一:稀疏数组 队列 链表
  12. ubuntu11.10 华为无线上网卡e303s
  13. 路由器连接校园网并发WIFI:WR703N路由器安装OpenWRT并运行H3C客户端操作步骤(主要针对中山大学东校区)
  14. python外星人入侵游戏图片_[Python]简单的外星人入侵游戏
  15. (java)scanner.next()与scanner.nextLine()的区别
  16. HCIA脱产班 学习笔记4
  17. 自己动手写一个简单的bootloader
  18. 二本 计算机专业2017分数线,2017全国大学二本录取分数线一键查询软件
  19. 记一次小米8从miui12降级miui10 解决fastboot模式miflash不识别问题
  20. 试玩R语言中教育诊断的那些包(一):kst

热门文章

  1. 软考信息系统监理师第五次作业
  2. 没有动任何配置文件,今天就出现了修改的JSP内容在页面不体现。依然是老的页面内容...
  3. sap idoc techniques
  4. 用DataTable.Merge()解决了一个排序问题
  5. Y/C分离/2/3D滤波器
  6. Xvid编码器流程(基于xvid1.1.0)
  7. QtCreator5.12.6安装图文教程
  8. Geometric-Transformations图像几何变换halcon算子,持续更新
  9. HALCON示例程序autobahn高速公路车道识别程序剖析
  10. orb-slam2在PC和ARM上运行