Android开发经验谈Android开发程序员

如何使用 Fragment 实现MVP架构

2019-04-12  本文已影响4人  Android小调

what?还能这样?

Fragment

早些时候,依赖 Activity 生命周期的操作以及业务逻辑都集中在 Activity 中,Activity 变成了单个页面的上帝类,大多数的代码都写在 Activity 中。而 Fragment 的出现分担 Activity 的重任,它和 Activity 有着同步的生命周期,它可以装载一系列 View 并管理这些 View 的生命周期,而一个 Activity 中可以包含多个 Fragment,这就意味着可以将不同功能布局的 View 用 Fragment 分开管理出来,这大大地减轻了 Activity 的负担,还从另一方面提供了适配屏幕大小以及横竖屏的方法。

但拥有与 Activity 同步生命周期的 Fragment 不单单可以用来装载 View,你也能将原先 Activity 中依赖生命周期的逻辑代码迁移到 Fragment 下,它比起 Activity 更加小巧灵活,而且被设计为完全可以不包含任何 View。

补充

我在知乎上一个回答上提到使用 Fragment(碎片)来进行重构,所以补充一下:

它能在重构中能做些什么?

细化视图/布局

细化逻辑(Activity 中的逻辑)

无限细化

无限细化是什么意思?就是你任何视图或者逻辑都可以用 Fragment 一直细化到你喜欢的粒度为止。一个 Fragment 内可以只有一个 View,也可以只用来实现一个后台下载任务。你应该懂为什么 Google cheng它为碎片了吧?

MVP

假设进行如下尝试:用没有 UI 的 Fragment 来构建 MVP 架构中的 Presenter,用来存放程序中的控制逻辑(调用业务逻辑代码,DO 到 VO 的转换等操作),它能带来以下一些明显好处:

能够同步 Activity 的生命周期,在生命周期内进行逻辑操作

能够很好地处理系统重建视图时(例如屏幕旋转)的现场恢复问题

它也能够在系统重建视图时不销毁实例,保留子属性,并且不打断正在进行的任务。(setRetainInstance(true))

这个 Presenter 看起来应该是这样的:

classMeiziPresenter():BasePresenter<Contract.View>(),Contract.Presenter{overridefunonViewCreated(view:Contract.View?,savedInstanceState:Bundle?){view?.showToast("View created.")}}

无疑这是一种新的尝试,我有看过类似的观点,但是并没有具体的实现。于是我在 Kotgo的新版本中将原有 Presenter 改为用 Fragment 实现。它一下子解决了我很多的问题。例如:

如何实现在屏幕旋转时依旧保持 Presenter 的实例

不保存 Presenter 实例的话,如何便捷地保存某些属性

如何在 Presenter 中自动同步 View 的生命周期(例如我要在 onCreate() 时做一些操作)

Fragment 的一些知识

对 NestedFragment 的 findFragmentByTag() 必需在 ParentFragment 的 onViewCreated()(视图创建后)中进行,否则将返回 null 值。

关于如何恢复视图以及数据现场网络上的文章很多有坑,下面是引用自我的 Git Blog,比较靠谱的一些笔记:

自定义 View 时,请使用 onSaveInstanceState() 和 onRestoreInstanceState() 处理视图状态的储存和恢复,以应付屏幕旋转等状况后视图的现场还原。

Fragment 在发生屏幕旋转等状况后,系统会持久化它的一些视图以及数据状态。旋转后 FragmentManager 会反系列化旋转前持久化的信息,新建实例,并在新实例的 onCreate() 中返回之前储存的各种 State(Fragment.onSaveInstanceState() 中插入的)。而 View State 会自动传递到各个 View 的View.onRestoreInstanceState() 函数中。

如果在 Fragment 中使用了 setRetainInstance(true),则 Fragment 的实例会被保留下来,不重新创建,这意味着实例内的所有属性也会被保存下来(不会被重置),但是依然会重新触发 Fragment 的生命周期事件。所以通常这种状况仅适用于进行持续性后台任务的 Fragment(例如没有视图的单纯进行下载操作的 Fragment),在屏幕旋转后也不会打断正在进行的任务。要注意的是,这种情况下如果有视图的话,视图会被重新创建,处理不好可能产生泄露。

附:架构学习资料

学习资料群:4112676    

群内:包含IOC,注入知识,rxjava2框架等

上一篇 下一篇

猜你喜欢

热点阅读