掌握Android Fragment开发之魂:Fragment的深度解析(下)

news/2024/5/20 14:39:02

在上一篇文章中,我们深入探讨了Fragment 通信,包含Fragment 向 Activity 传递数据、Activity 向 Fragment 传递数据、Fragment 之间的通信方式。感兴趣的朋友,请前往查阅: 掌握Android Fragment开发之魂:Fragment的深度解析(中) 。


在这篇文章中,我们将继续深入探讨 Android Fragment,包括以下几个核心主题:

  • DialogFragment 是什么

  • DialogFragment相比 Dialog的优势

  • DialogFragment 的使用方式

  • Fragment 在 Android 开发中的最佳实践


一、DialogFragment 的魅力所在

1、DialogFragment 是什么?

DialogFragment 是 Android 开发中用于创建和管理对话框的一个类,它是 Fragment 的一个子类,专门用于显示对话框界面。在 Android 中,对话框是一种浮动的 UI 元素,通常用于提供额外的信息或选项,而不会干扰用户当前的操作。

以下是 DialogFragment 的一些关键特点:

  1. 模态性DialogFragment 通常以模态的形式出现,意味着用户必须与对话框交互(如确认或取消操作)才能返回到应用程序的其他部分。

  2. 生命周期DialogFragment 拥有自己的生命周期,与宿主 Activity 的生命周期相互独立。它在 onCreate 方法中初始化,在 onCreateView 方法中创建视图,在 onShow 方法中显示,在 onDismiss 方法中关闭。

  3. 自定义视图:开发者可以在 onCreateView 方法中定义自己的布局文件,从而实现高度自定义的对话框界面。

  4. 显示与隐藏:可以通过调用 show 方法将 DialogFragment 显示在宿主 Activity 上,也可以调用 dismiss 方法将其隐藏。

  5. 主题和样式:可以为 DialogFragment 设置不同的主题和样式,以适应不同的 UI 设计需求。

  6. 交互DialogFragment 可以包含按钮、列表、文本框等 UI 组件,允许用户进行交互。

  7. BackStack:当 DialogFragment 被添加到 FragmentTransaction 时,可以选择是否将其添加到回栈(BackStack)中,这允许用户通过按返回键来导航。


2、DialogFragment相比 Dialog的优势


DialogFragment 相比于传统的 Dialog 有以下几个优势:

  • 生命周期可控

    DialogFragment拥有自己的生命周期,能够更好地与ActivityFragment的生命周期进行协同。当Activity被销毁时,与之关联的DialogFragment` 也会被销毁,这有助于避免内存泄漏。

  • UI更新

    DialogFragment` 可以更容易地响应配置更改(如屏幕旋转),在这些情况下,对话框可以保持其状态,而不需要重新创建。

  • BackStack 管理

    DialogFragment 可以被添加到 Fragment 的回栈(BackStack)中,这意味着用户可以使用返回键来关闭对话框,提供了更好的用户体验。

  • 样式和主题

    DialogFragment 允许开发者通过设置不同的主题和样式来自定义对话框的外观,这在传统的 Dialog 中实现起来可能更加困难。

  • 兼容性

    随着 Android 系统的更新,DialogFragment 能够更好地适应新的 UI 设计和交互模式,而传统的 Dialog 可能需要更多的工作来保持兼容性。

  • 代码复用,解耦设计

    DialogFragment 允许开发者创建可复用的对话框模板,可以在不同的 Activity 中重复使用,而不需要为每个 Activity 编写相同的代码。

  • 与 Fragment 协同

    如果你的应用程序已经使用了 Fragment 来管理 UI 的不同部分,使用 DialogFragment 可以保持代码的一致性,并且可以更容易地在 Fragment 之间共享数据。

  • 动画和转场支持

    DialogFragment` 支持自定义的进入和退出动画,这使得对话框的显示和隐藏可以有更丰富的视觉效果。

  • 更清晰的代码结构

    使用 DialogFragment 可以使代码结构更加清晰,因为每个对话框都是一个独立的实体,拥有自己的逻辑和布局。

  • 支持 Fragment 事务

    DialogFragment` 可以作为 Fragment 事务的一部分进行管理,这使得在运行时添加、替换或删除对话框变得更加容易。

  • 丰富的回调

    DialogFragment 提供了更多的回调方法,如 onCreateDialog()onResume() 等。


总的来说,DialogFragment 提供了更好的生命周期管理、更灵活的 UI 更新、更丰富的自定义选项以及更好的用户体验,这些优势使其成为在 Android 应用中创建和管理对话框的首选方式。


3、DialogFragment 的使用方式

接下来,我们通过一个示例来演示 DialogFragment 的使用方式。

public class MyDialogFragment extends DialogFragment {@NonNull@Overridepublic Dialog onCreateDialog(@Nullable Bundle savedInstanceState) {AlertDialog.Builder builder = new AlertDialog.Builder(getActivity());builder.setTitle("标题").setMessage("这是一个 DialogFragment").setPositiveButton("确定", new DialogInterface.OnClickListener() {@Overridepublic void onClick(DialogInterface dialog, int which) {// 处理确定按钮点击事件}}).setNegativeButton("取消", new DialogInterface.OnClickListener() {@Overridepublic void onClick(DialogInterface dialog, int which) {// 处理取消按钮点击事件}});return builder.create();}
}

要显示该 DialogFragment,只需要在 Activity 中执行以下代码:

public class MainActivity extends AppCompatActivity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);// 显示 DialogFragmentshowDialogFragment();}private void showDialogFragment() {MyDialogFragment dialogFragment = new MyDialogFragment();dialogFragment.show(getSupportFragmentManager(), "MyDialogFragment");}
}

MainActivity 中,我们首先实例化 MyDialogFragment。然后,调用 show() 方法并传入 FragmentManager 实例和一个可选的标签(用于标识该 DialogFragment)来显示该 DialogFragment。

show() 方法的完整签名如下:

public void show(@NonNull FragmentManager manager, @Nullable String tag)
  • manager: 用于管理 Fragment 的 FragmentManager 实例。
  • tag: 可选的字符串标签,用于标识该 DialogFragment。如果你需要在将来获取该 DialogFragment 的实例,可以使用这个标签。

在上面的示例中,我们使用 getSupportFragmentManager() 获取 FragmentManager 实例,并将标签设置为 “MyDialogFragment”。

你还可以通过 DialogFragment 的其他方法来自定义对话框的样式和行为,例如:

  • setStyle(): 设置对话框的样式(默认、全屏等)。
  • setCancelable(): 设置对话框是否可以通过点击外部区域或按下返回键来取消。
  • setShowsDialog(): 控制是否显示对话框标题和按钮。

通过上述代码和扩展方法,你就可以灵活地创建和管理 DialogFragment 了。DialogFragment 不仅提供了标准的 Fragment 生命周期管理,还增强了对话框的可定制性,是展示模态界面的绝佳选择。


二、Fragment 在 Android 开发中的最佳实践

在 Android 应用程序开发中,合理使用 Fragment 可以带来诸多好处,提高代码的可维护性和用户体验。因此,掌握 Fragment 的最佳实践是非常有必要的。


1、保持 Fragment 精简

Fragment 被设计为可重用的界面构建块,因此应该保持其职责单一、逻辑精简。

将过多的业务逻辑耦合到 Fragment 中会增加其复杂性,影响可维护性。

相反,应该将业务逻辑分离到其他层次,如 ViewModel 或 Repository,而让 Fragment 专注于界面呈现和用户交互。


2、高度解耦设计

遵循单一责任原则,Fragment 应该与其他组件(如 Activity、ViewModel 等)高度解耦。

这种解耦设计可以最大限度地提高代码的可复用性和可测试性。

例如,可以通过定义接口或使用依赖注入等方式来降低 Fragment 与其他组件的耦合度。


3、避免过度嵌套

虽然 Fragment 可以嵌套,但过度嵌套会导致界面层次过深,增加复杂性和性能开销。

因此,应该权衡利弊,尽量保持 Fragment 嵌套的层次浅一些。

如果确实需要嵌套,可以考虑使用 ViewPager 等容器来管理多个 Fragment。


4、注意生命周期管理

Fragment 具有自己的生命周期,与其宿主 Activity 的生命周期存在一定的联系。

在编写 Fragment 代码时,需要特别注意生命周期方法的调用顺序和时机,避免出现内存泄漏或数据丢失等问题。

同时,也要注意在适当的生命周期方法中执行相应的操作,如数据加载、视图绑定等。


public class MyFragment extends Fragment {@Overridepublic void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);// 初始化操作}@Overridepublic View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {// 加载布局return inflater.inflate(R.layout.fragment_my, container, false);}@Overridepublic void onViewCreated(View view, Bundle savedInstanceState) {super.onViewCreated(view, savedInstanceState);// 初始化视图组件}@Overridepublic void onDestroyView() {super.onDestroyView();// 清理操作,如取消动画、移除监听器等}@Overridepublic void onDestroy() {super.onDestroy();// 最终清理操作}
}

5、利用 FragmentManager 的力量


FragmentManager 是管理 Fragment 生命周期和事务的核心组件。

掌握 FragmentManager 的使用技巧,如 add()replace()remove()addToBackStack() 等方法,可以更好地控制 Fragment 的行为,可以确保 UI 更新的一致性。

同时,也要注意合理利用回退栈,为用户提供流畅的导航体验。

FragmentTransaction transaction = getActivity().getSupportFragmentManager().beginTransaction();
transaction.replace(R.id.fragment_container, new MyFragment()); // 替换 Fragment
transaction.addToBackStack(null); // 添加到回栈
transaction.commit(); // 提交事务

6、避免在 Fragment 中进行长时间运行的操作

长时间运行的操作应该在后台线程中执行,以避免阻塞主线程。

new AsyncTask<Void, Void, Void>() {@Overrideprotected Void doInBackground(Void... voids) {// 长时间运行的操作return null;}@Overrideprotected void onPostExecute(Void aVoid) {// 更新 UI}
}.execute();

7、使用 setUserVisibleHint 来优化性能

这个方法可以帮助了解 Fragment 是否对用户可见,从而优化性能。

@Override
public void setUserVisibleHint(boolean isVisibleToUser) {super.setUserVisibleHint(isVisibleToUser);if (isVisibleToUser) {// 当 Fragment 对用户可见时执行的操作}
}

8、正确处理 Fragment 回栈

当用户按下返回键时,确保正确处理 Fragment 回栈,提供一致的用户体验。

@Override
public void onBackStackChanged() {super.onBackStackChanged();if (getFragmentManager().getBackStackEntryCount() == 0) {// 处理回栈为空的情况,如退出 Activity}
}

9、使用 FragmentnewInstance 方法

这是一种创建 Fragment 实例的推荐方式,它使得传递参数变得更加容易。

复制
public static MyFragment newInstance() {MyFragment fragment = new MyFragment();Bundle args = new Bundle();// 向 args 中添加参数fragment.setArguments(args);return fragment;
}

10、避免内存泄漏

确保在 onDestroyonDestroyView 中清理资源,如移除监听器、取消动画等。

@Override
public void onDestroyView() {super.onDestroyView();if (someListener != null) {someView.removeOnClickListener(someListener);someListener = null;}
}

11、使用 LoaderManager 进行数据加载

使用 LoaderManager 可以帮助你管理后台数据加载,并在 Fragment 重新创建时恢复加载状态。

@Override
public void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);getLoaderManager().initLoader(0, null, this);
}@Override
public Loader<Cursor> onCreateLoader(int id, Bundle args) {// 创建并返回数据加载器
}@Override
public void onLoadFinished(Loader<Cursor> loader, Cursor data) {// 数据加载完成,更新 UI
}@Override
public void onLoaderReset(Loader<Cursor> loader) {// 数据加载器被重置,清理数据
}

通过上述最佳实践,我们能够更高效地管理 Fragment,提升应用的稳定性和性能,可以帮助你创建更加健壮且易于维护的 Android 应用。



http://www.mrgr.cn/p/13168158

相关文章

Java面试八股文(SpringCloud篇)

****************************************************

Error: Cannot find module ‘D:\SoftSetupLoaction\nodejs\node_global\node_modules\npm\bin\npm-cli.js‘

Error: Cannot find module ‘D:\SoftSetupLoaction\nodejs\node_global\node_modules\npm\bin\npm-cli.js‘ 出现原因: 重新安装可装了nodejs和npm 网上查了很多方法,都建议重装,但是都没有效果(因为我就是重装之后出现的问题) 按照错误提示node_global找不到npm-cli.js,个人…

初探pinctrl子系统和GPIO子系统

前言: 在前面的led驱动程序和按键驱动程序中,无论是最传统的方法,还是总线设备驱动模型,还是基于设备树的总线设备驱动模型,都是直接操作寄存器的方法。驱动开发的本质确实是操作寄存器,但是一个芯片有几百个引脚,只是操作少数的几个引脚还好,如果是大量的引脚,比如LC…

前端开发攻略---使用Sass调整颜色亮度,实现Element组件库同款按钮

目录 1、演示 2、实现原理 3、实现代码 1、演示 2、实现原理 改变颜色亮度的原理是通过调整颜色的 RGB 值中的亮度部分来实现的。在 Sass 中&#xff0c;可以使用颜色函数来操作颜色的 RGB 值&#xff0c;从而实现亮度的调整。 具体来说&#xff0c;亮度调整函数通常会改变颜…

PVE新增硬盘并扩容给 local分区

PVE安装在120G的固态硬盘,现在加了一块1T的机械硬盘作为虚拟机系统用,需要把磁盘扩容给 local分区 1、ssh连上pve,使用 lsblk 查看硬盘驱动器路径,我这里新加的硬盘是 sda,硬盘还未进行分区 2、fdisk /dev/sda,对硬盘进行分区操作,注意你自己的硬盘名称,千万小心不要搞…

Windows内核开发:如何使用STL

前言 大家都知道应用层c的STL非常强大&#xff0c;非常好用&#xff0c;但是在内核下就没法用了。针对这个问题&#xff0c;经过我不懈的寻找&#xff0c;终于找到了解决内核无法使用STL的方法。 使用new/delete关键字 先说一下常用关键字如何在内核中使用。其实只需要在一个全…

实验1-波士顿房价预测部分报错解决方法

运行sgd = SGDRegressor()sgd.fit(x_train, y_train)print("r2 score of Linear regression is",r2_score(y_test,sgd.predict(x_test)))时出现 DataConversionWarning: A column-vector y was passed when a 1d array was expected. Please change the shape of y t…

用栈实现队列——leetcode刷题

题目要求我们只用栈的基本操作 push to top 入栈&#xff0c;peek from top 返回栈顶元素&#xff0c;pop from top 移除并返回栈顶元素&#xff0c;size 栈的大小&#xff0c;is_empty 判断栈是否为空&#xff0c;这几个函数来实现队列&#xff0c;也就是说&#xff0c;我们在…

【Java】初识网络编程

文章目录 前言✍一、互联网的发展1.独立模式2.网络的出现局域网LAN广域网WAN ✍二、网络编程概述✍三、网络编程中的术语介绍IP地址端口号协议OSI七层模型TCP\IP四层模型 ✍四、协议的层级之间是如何配合工作的 前言 在本文中&#xff0c;会对网络编程的一些术语进行解释&#…

《最新出炉》系列入门篇-Python+Playwright自动化测试-45-鼠标操作-下篇

1.简介 鼠标为我们使用电脑提供了很多方便,我们看到的东西就可以将鼠标移动过去进行点击就可以打开或者访问内容,当页面内容过长时,我们也可以使用鼠标滚轮来实现对整个页面内容的查看,其实playwright也有鼠标操作的方法。上一篇文章中已经讲解过鼠标的部分操作了,今天宏哥…

redux中核心组件有哪些,reducer的作用

在redux中,核心组件包括Action、Reducer、Store和Middleware。 Action是一个普通的JavaScript对象,用于描述发生了什么事件。它必须包含一个type属性,用于标识事件的类型。可以在Action中添加其他自定义的属性来传递数据。 Reducer是一个纯函数,用于根据收到的Action来更新…

学习记录+vcode+GPIO例程+正点原子 DNESP32S3 开发板教程-IDF 版

第一个程序: UART模式和JTAG模式,配置完成不同。配置主要就是.vscode 文件夹中 c_cpp_properties.json,tasks.json,launch.json,settings.json四个文件。 一个想法:备份UART模式和JTAG模式的配置文件,用时直接文件替换。简单粗暴方式是.vscode 文件夹替换。当然每次要选…

chrome extension插件替换网络请求中的useragent

感觉Chrome商店中的插件不能很好的实现自己想要的效果,那么就来自己动手吧。 本文以百度为例: 一般来说网页请求如下: 当前使用的useragent是User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/124.0.0.0 Safar…

AB实验相关流程

本篇文章介绍的是一个完整AB测试流程应该怎么走。 AB测试流程有以下几个步骤: 一、选取实验指标 二、建立实验假设 三、选取实验单位 四、确定最小提升预期值 五、计算最小样本量 六、流量分割 七、确定实验时长 八、数据统计 九、得出结论 接下来就详细说明每个步骤。 一、选…

RS232引脚方向及意义与接线参考

RS232引脚方向及定义编号引脚意义方向作为IO使用说明1CD载波检测(Carrier Detect)计算机《调制解调器输入调制解调器通知计算机有载波被侦测到2RXD接收(Receive)计算机《调制解调器 接收数据3TXD发送(Transmit)计算机》调制解调器 发送数据4DTR数据终端设备(DTE)备好(Data Te…

jenkins 拉取代码之后 自动执行jar包到部署服务器自动运行

原文地址: https://blog.csdn.net/xiuyuandashen/article/details/124490378

springboot整合rabbitmq的不同工作模式详解

前提是已经安装并启动了rabbitmq&#xff0c;并且项目已经引入rabbitmq&#xff0c;完成了配置。 不同模式所需参数不同&#xff0c;生产者可以根据参数不同使用重载的convertAndSend方法。而消费者均是直接监听某个队列。 不同的交换机是实现不同工作模式的关键组件.每种交换…

MindSpore反向传播配置关键字参数

继上一篇文章从Torch的两个Issue中找到一些类似的问题之后,可以发现深度学习框架对于自定义反向传播函数中的传参还是比较依赖于必备参数,而不是关键字参数,MindSpore深度学习框架也是如此。但是我们可以使用一些临时的解决方案,对此问题进行一定程度上的规避,只要能够自定…

AR精灵——风险分析和典型用户

风险分析典型用户 典型用户一名字:盛宇伟 年龄:28岁,收入:每月约8000元 代表的用户在市场上的比例和重要性:虽然使用AR精灵的付费用户比例较少,但他们对产品的热爱和忠诚度很高,他们的反馈和建议对产品的改进至关重要。 使用这个软件的典型场景:李梅在下班后回到家中,…