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


在上一篇文章中,我们深入探讨了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.mzph.cn/pingmian/9053.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

mkdocs工作及备份流

欢迎光顾泥烟の新站 本文链接 &#x1f449;mkdocs工作及备份流 - 泥烟 (knight02-bit.github.io) mkdocs 工作及备份流 为了防止鸽太久忘记工作流了,遂简化流程并备份于此 目录结构 KnightDESKTOP-31LJ6LM MINGW64 /k/blog_mkdocs $ tree -L 3 -d . |-- articles(草稿) | …

QpushButton实现文本自动换行

1. 第一种 可以使用布局管理器或者设置按钮的最小宽度和固定高度。 以下是一种通过布局管理器实现QPushButton按钮文本自动换行的方法&#xff1a; #include <QApplication> #include <QPushButton> #include <QVBoxLayout>int main(int argc, char *argv[])…

docker学习-docker常用其他命令整理

随便写写&#xff0c;后面有空再更新 镜像命令&#xff0c;容器命令已在之前略有更新&#xff0c;这次不写&#xff0c; 一、后台启动命令 # 命令 docker run -d 容器名 # 例子 docker run -d centos # 启动centos&#xff0c;使用后台方式启动 # 问题&#xff1a; 使用doc…

大数据手册(Spark)--Spark 简介

Spark 简介 Apache Spark 是一种用于大数据工作负载的分布式开源处理系统。它使用内存中缓存和优化的查询执行方式&#xff0c;可针对任何规模的数据进行快速分析查询。Apache Spark 提供了简明、一致的 Java、Scala、Python 和 R 应用程序编程接口 (API)。 Apache Spark 是专…

代码随想录第四十三天|最后一块石头的重量 II 、目标和

题目链接&#xff1a;. - 力扣&#xff08;LeetCode&#xff09; 代码如下&#xff1a; 题目链接&#xff1a;. - 力扣&#xff08;LeetCode&#xff09; 代码如下&#xff1a;

QpushButton绘制圆角矩形并绘制背景颜色

要在QPushButton的paintEvent中绘制按钮的颜色&#xff0c;可以创建一个自定义的QPushButton类&#xff0c;并重写它的paintEvent函数。在paintEvent函数中&#xff0c;可以使用QPainter来绘制按钮的背景颜色。 以下是一个简单的示例代码&#xff0c;演示如何在自定义的QPushB…

用户行为分析与内容创新:Kompas.ai的数据驱动策略

在数字化营销的今天&#xff0c;用户行为数据分析已成为内容创新和策略调整的核心。通过深入理解用户的行为模式和偏好&#xff0c;品牌能够创造出更具吸引力和相关性的内容&#xff0c;从而实现精准营销。本文将探讨用户行为数据分析在内容创新和策略调整中的价值&#xff0c;…

【Linux】进程间通信方式之管道

&#x1f916;个人主页&#xff1a;晚风相伴-CSDN博客 &#x1f496;如果觉得内容对你有帮助的话&#xff0c;还请给博主一键三连&#xff08;点赞&#x1f49c;、收藏&#x1f9e1;、关注&#x1f49a;&#xff09;吧 &#x1f64f;如果内容有误的话&#xff0c;还望指出&…

伪类和伪元素的区别是什么?

一、两者的定义 1.伪类&#xff08;pseudo-class&#xff09;是一个以冒号作为前缀&#xff0c;被添加到一个选择器末尾的关键字&#xff0c;当你希望样式在特定状态才被呈现到指定的元素时&#xff0c;你可以往元素的选择器后面加上对应的伪类。 2.伪元素用于创建一些不在文档…

【C++ | 语句】条件语句(if、switch)、循环语句(while、do while、for、范围for)、跳转语句、try语句块和异常处理

&#x1f601;博客主页&#x1f601;&#xff1a;&#x1f680;https://blog.csdn.net/wkd_007&#x1f680; &#x1f911;博客内容&#x1f911;&#xff1a;&#x1f36d;嵌入式开发、Linux、C语言、C、数据结构、音视频&#x1f36d; ⏰发布时间⏰&#xff1a;2024-05-02 2…

k8s部署skywalking(helm)

官方文档 官方文档说明&#xff1a;Backend setup | Apache SkyWalking官方helm源码&#xff1a;apache/skywalking-helm官方下载&#xff08;包括agent、apm&#xff09;:Downloads | Apache SkyWalking 部署 根据官方helm提示&#xff0c;选择你自己部署的方式&#xff0c…

Petalinux的使用——定制Linux系统

文章目录 配置petalinux运行环境petalinux设计流程 配置petalinux运行环境 Petalinux的安装在文章Ubuntu镜像源的更改及其Petalinux的安装中已经介绍&#xff0c;下面介绍petalinux运行环境的配置过程。 进入到petalinux的安装路径下&#xff0c;使用下面的命令对petalinux的运…

C语言洛谷题目分享(11)回文质数

目录 1.前言 2.题目&#xff1a;回文质数 1.题目描述 2.输入格式 3.输出格式 4.输入输出样例 5.题解 3.小结 1.前言 哈喽大家好&#xff0c;今儿继续为大家分享一道蛮有价值的一道题&#xff0c;希望大家多多支持喔~ 2.题目&#xff1a;回文质数 1.题目描述 因为 151 …

【用文本生成歌声】Learn2Sing 2.0——歌声转换算法即梅尔频谱详解

一. 频谱图与梅尔谱图的介绍 频谱图&#xff1a;频谱图可以理解为一堆垂直堆叠在一起的快速傅里叶变换结果。 1.1 信号 在进入频谱图模块之前&#xff0c;首先我们需要了解信号是什么。 信号就是某一特定量随时间变化&#xff0c;对于音频来说&#xff0c;这个特定的变化量就…

pytest教程-43-钩子函数-pytest_report_header

领取资料&#xff0c;咨询答疑&#xff0c;请➕wei: June__Go 上一小节我们学习了pytest_runtest_makereport钩子函数的使用方法&#xff0c;本小节我们讲解一下pytest_report_header钩子函数的使用方法。 pytest_report_header 钩子函数允许你在 pytest 的终端报告的头部添…

七、 数据出境安全评估申报需要多长时间?

《评估申报指南&#xff08;第二版&#xff09;》未区分数据处理者进行数据出境安全评估线上申报和线下申报整体所需时间。一般情况下&#xff0c;数据出境安全评估的申报时长周期如图所示&#xff1a; 根据《评估申报指南&#xff08;第二版&#xff09;》第二条的规定&#…

python开发二

python开发二 requests请求模块 requests 是一个常用的 Python 第三方库&#xff0c;用于发送 HTTP 请求。它提供了简洁且易于使用的接口&#xff0c;使得与 Web 服务进行交互变得非常方便。 发送 GET 请求并获取响应 import requestsresponse requests.get("https:/…

跨越智能建筑桥梁:西门子PLC无缝对接BACnet楼宇自动化系统化

智能楼宇每一个环节的互联互通都至关重要&#xff0c;而PLC&#xff08;可编程逻辑控制器&#xff09;作为自动化领域的基石&#xff0c;其与BACnet协议的融合无疑成为了构建智能楼宇神经系统的关键节点。今天&#xff0c;让我们深入探讨如何利用先进的PLC转BACnet协议网关&…

使用Python实现2048小游戏

使用Python实现2048小游戏源码分享。实现效果如下所示。 实现效果图 游戏开始效果图 游戏结束效果图 部分源码截图 下载链接 基于如下的运行环境。运行需要安装tkinter /Library/Frameworks/Python.framework/Versions/3.7/bin/python/bin/python /Users/nihui/Documents/P…

AI预测体彩排3第3套算法实战化赚米验证第1弹2024年5月5日第1次测试

从今天开始&#xff0c;准备启用第3套算法&#xff0c;来验证下本算法的可行性。因为本算法通过近三十期的内测&#xff08;内测版没有公开预测结果&#xff09;&#xff0c;发现本算法的预测结果优于其他所有算法的效果。彩票预测只有实战才能检验是否有效&#xff0c;只有真正…