Fragment详解

简介: 前言谢谢你那么的、安静的捧场。Fragment的概述及其设计初衷Fragment代表了Activity的子模块,因此可以把Fragment理解成Activity片段。

前言

谢谢你那么的、安静的捧场。

Fragment的概述及其设计初衷

Fragment代表了Activity的子模块,因此可以把Fragment理解成Activity片段。Fragment必须被“嵌入”Activity中使用,因此,虽然Fragment也拥有自己的生命周期,但Fragment的生命周期会受到它所在的Activity的生命周期控制。只有当该Activity处于活动状态时,程序员才可通过方法独立操作Fragment。
关于Fragment,可以归纳出如下几个特征。
  • Fragment总是作为Activity界面的组成部分。Fragment可调用getActivity()方法获取它所在的Activity,Activity可调用FragmentManager的findFragmentById()或findFragmentByTag()方法来获取Fragment。

  • 在Activity运行过程中,可调用FragmentManager的add()、remove()、replace()方法动态地添加、删除或替换Fragment。

  • 一个Activity可以同时组合多个Fragment;反过来,一个Fragment也可被多个Activity复用。

  • Fragment可以响应自己的输入事件,并拥有自己的生命周期,但它们的生命周期直接被其所属的Activity的生命周期控制。

Fragment的初衷是为了适应大屏幕的平板电脑,由于平板电脑的屏幕比手机屏幕更大,因此可以容纳更多的UI组件,且这些UI组件之间存在交互关系。Fragment简化了大屏幕UI的设计,它不需要开发者管理组件包含关系的复杂变化,开发者使用Fragment对UI组件进行分组、模块化管理,就可以更方便地在运行过程中动态更新Activity的用户界面。

Fragment的生命周期

与Activity类似的是,Fragment也存在如下状态。
  • 运行状态:当前Fragment位于前台,用户可见,可以获得焦点。

  • 暂停状态:其他Activity位于前台,该Fragment依然可见,只是不能获得焦点。

  • 停止状态:该Fragment不可见,失去焦点。

  • 销毁状态:该Fragment被完全删除,或该Fragment所在的Activity被结束。

下图显示了Fragment的生命周期及相关回调方法。
img_709a941ab1a9e277eff84a3f1e71a1e7.png
lifecycle of fragment.png
从图中可以看出,在Fragment的生命周期中,如下方法被系统回调。
  • onAttach():当该Fragment被添加到Activity时被回调。该方法只会被调用一次。

  • onCreate():创建Fragment时被调用。该方法只会被调用一次。

  • onCreateView():每次创建、绘制该Fragment的View组件时回调该方法,Fragment将会显示该方法返回的View组件。

  • onActivityCreated():当Fragment所在的Activity被启动完成后回调该方法。

  • onStart():启动Fragment时被调用。

  • onResume():恢复Fragment时被回调,在OnStart()方法后一定会调用onResume()方法。

  • onPause():暂停Fragment时被回调。

  • onStop():停止Fragment时被回调。

  • onDestroyView():销毁该Fragment所包含的View组件时调用。

  • onDestroy():销毁该Fragment时被毁掉。该方法只会调用一次。

  • onDetach():将该Fragment从Activity中删除、替换完成时回调该方法,在onDestroy()方法后一定会回调onDetach()方法。该方法只会被调用一次。

代码示例

LifecycleFragment.java
public class LifecycleFragment extends Fragment {

    final String TAG = "--halo--";

    @Override
    public void onActivityCreated(Bundle savedInstanceState) {
        super.onActivityCreated(savedInstanceState);
        Log.d(TAG, "---onActivityCreated---");
    }

    @Override
    public void onAttach(Activity activity) {
        super.onAttach(activity);
        Log.d(TAG, "---onAttach---");
    }

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        Log.d(TAG, "---onCreate---");
    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        Log.d(TAG, "---onCreateView---");

        TextView tv = new TextView(getActivity());
        tv.setGravity(Gravity.CENTER_HORIZONTAL);
        tv.setText("测试Fragment");
        tv.setTextSize(40);
        return tv;
    }

    @Override
    public void onDestroy() {
        super.onDestroy();
        Log.d(TAG, "---onDestroy---");
    }

    @Override
    public void onDestroyView() {
        super.onDestroyView();
        Log.d(TAG, "---onDestroyView---");
    }

    @Override
    public void onDetach() {
        super.onDetach();
        Log.d(TAG, "---onDetach---");
    }

    @Override
    public void onPause() {
        super.onPause();
        Log.d(TAG, "---onPause---");
    }

    @Override
    public void onResume() {
        super.onResume();
        Log.d(TAG, "---onResume---");
    }

    @Override
    public void onStart() {
        super.onStart();
        Log.d(TAG, "---onStart---");
    }

    @Override
    public void onStop() {
        super.onStop();
        Log.d(TAG, "---onStop---");
    }
}

LifecycleFgActivity.java
public class LifecycleFgActivity extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.lifecycle);

        LifecycleFragment fragment = new LifecycleFragment();
        FragmentTransaction transaction = getFragmentManager().beginTransaction();
        transaction.replace(R.id.life_cycle, fragment).commit();

    }

    public void bt0(View v) {
        Intent intent = new Intent(this, DialogActivity.class);
        startActivity(intent);
    }

    public void bt1(View v) {
        LifecycleFragment fragment = new LifecycleFragment();
        FragmentTransaction transaction = getFragmentManager().beginTransaction();
        transaction.replace(R.id.life_cycle, fragment);
        transaction.addToBackStack(null);
        transaction.commit();

    }

    public void bt2(View v) {
        LifecycleFragment fragment = new LifecycleFragment();
        FragmentTransaction transaction = getFragmentManager().beginTransaction();
        transaction.replace(R.id.life_cycle, fragment);
        transaction.commit();
    }
}
lifecycle.xml
<?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" >

    <Button
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:onClick="bt0"
        android:text="打开对话框" />

    <Button
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:onClick="bt1"
        android:text="替换目标Fragment,并加入Back栈" />

    <Button
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:onClick="bt2"
        android:text="退出" />
      <FrameLayout
        android:id="@+id/life_cycle"
        android:layout_width="match_parent"
        android:layout_height="match_parent" />
</LinearLayout>

效果

当使用Activity加载该Fragment时,可以在LogCat控制台看到下图的情况。
img_ed8a19fb3f6d2937731959789cf39d07.png
lifecycle1.PNG
如果单击程序中的“打开对话框”按钮,将启动一个对话框风格的Activity,当前的Activity将会转入暂停状态,该Fragment已经进入“暂停”状态,此时可以在LogCat控制台看到下图的情况。
img_1cb9319943b88c2f1de0bff272f16e07.png
lifecycle2.PNG
关闭对话框风格的Activity,Fragment将会再次进入“运行”状态,将可以在LogCat控制台看到下图的情况。
img_7ee1acc5019cdff2608cdd59ef0e7ab1.png
lifecycle3.PNG
单击程序界面上的“替换目标Fragment,并加入Back栈”,将可以在LogCat控制台看到如下图的情况。
img_f78610de854f9e82fccc8379751954f5.png
lifecycle4.PNG
当你点击手机上的返回按钮时,该fragment将会再次显示出来,此时可以在LogCat控制台看到如下图的情况。
img_857874217e1905d81c14acad44e8d5cd.png
lifecycle5.PNG
单击程序界面上的“退出”按钮,该Fragment将会被完全结束,Fragment将进入“销毁”状态,此时可以在LogCat控制人看到如下图的情况。
img_41d93fbbbbd70da153f9db85a985bba7.png
lifecycle6.PNG
目录
相关文章
|
数据采集 自然语言处理 Devops
ToolLearning Eval:CodeFuse发布首个中文Function Call的大语言模型评测基准!🚀
CodeFuse发布了首个面向ToolLearning领域的中文评测基准ToolLearning-Eval,以帮助开发者跟踪ToolLearning领域大模型的进展,并了解各个ToolLearning领域大模型的优势与不足。ToolLearning-Eval按照Function Call流程进行划分,包含工具选择、工具调用、工具执行结果总结这三个过程,方便通用模型可以对各个过程进行评测分析。
1568 0
|
2月前
|
存储 监控 Cloud Native
吃透云原生可观测:Metrics、Logging、Tracing 架构底层逻辑与实战全指南
云原生可观测性是应对分布式系统复杂性的核心能力,以Metrics(指标)、Logging(日志)、Tracing(链路追踪)三大支柱为支撑,通过TraceId串联,实现故障快速定位、性能优化与根因分析。OpenTelemetry提供统一标准与自动埋点能力。
661 1
|
10月前
|
Web App开发 Linux Shell
CuteHTTPFileServer下载,局域网文件传输工具下载,chfs支持的最低SSL版本为SSLv3
FinalShell是一款支持多平台的SSH客户端工具,提供一体化服务器管理功能,支持shell与sftp同屏显示、命令自动提示和访问加速,操作简单高效。
948 12
sql面试50题------(1-10)
这篇文章提供了SQL面试中的前10个问题及其解决方案,包括查询特定条件下的学生信息、教师信息和课程成绩等。
sql面试50题------(1-10)
|
9月前
|
Ubuntu 安全 小程序
服务器版本的CentOS和Ubuntu哪个更适合你?
但是以上的比较并不说明Ubuntu是不稳定的或者是不安全的,只是以上比较过程中,在稳定性方面Ubuntu稍微逊色了一点。由于Ubuntu在个人桌面电脑的使用率远远高于CentOS,用Ubuntu搭建服务器,如果遇到什么问题,寻找解决方案相对比较容易,这让Ubuntu在选择方面更优于CentOS。如果你是一个初学者,那么毫无疑问Ubuntu是更适合的选择。如果你正在经营自己的公司,在这两者之间,CentOS会更好一些。
|
定位技术
域名前缀和后缀html,为什么域名前要加www前缀,www是什么意思?
为什么域名前要加www前缀?Michael F Liu号召大家把域名前面的www去掉,我深以为然。好域名都被瓜分光了,大家手里的域名都老长老长的,处理网域名[https://www.91chuli.com/](https://www.91chuli.com/)就有5个字母,前面再加上“www.”,多让直接访问者敲打5次键盘,何苦来呢?
18126 6
|
敏捷开发 测试技术 持续交付
阿里云云效产品使用合集之从GitHub下载代码失败是什么原因
云效作为一款全面覆盖研发全生命周期管理的云端效能平台,致力于帮助企业实现高效协同、敏捷研发和持续交付。本合集收集整理了用户在使用云效过程中遇到的常见问题,问题涉及项目创建与管理、需求规划与迭代、代码托管与版本控制、自动化测试、持续集成与发布等方面。
|
Python
如何使用正则表达式提取网页中的特定信息
如何使用正则表达式提取网页中的特定信息
769 1
|
SQL 存储 关系型数据库
Apache Flink 和 Paimon 在自如数据集成场景中的使用
自如目前线上有基于 Hive 的离线数仓和基于 Flink、Kafka 的实时数仓,随着业务发展,我们也在探索引入湖仓一体的架构更好的支持业务,我们对比了 Iceberg、Hudi、Paimon 后,最终选择 Paimon 作为我们湖仓一体的存储引擎,本文分享下自如在引入 Paimon 做数据集成的一些探索实践。
1410 1
Apache Flink 和 Paimon 在自如数据集成场景中的使用