微信号:jszj2014215

介绍:主要分享移动互联网的相关产品和资讯,关注你将学习到更多,在互联网的当下你会赚更多的钱...

Android使用 Fragment 构建 Presenter

2016-04-13 07:53 Android技术之家

作者:nekocode
链接:http://zhuanlan.zhihu.com/p/20656755
来源:知乎
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

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 看起来应该是这样的:

class MeiziPresenter(): BasePresenter() {
    interface ViewInterface {
        fun refreshMeizis(meizis: List<Meizi>)
    }

    var view: ViewInterface? = null
    override fun onAttach(activity: Activity?) {
        super.onAttach(activity)
        view = getParent() as ViewInterface
    }}

无疑这是一种新的尝试,我有看过类似的观点,但是并没有具体的实现。于是我在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),在屏幕旋转后也不会打断正在进行的任务。要注意的是,这种情况下如果有视图的话,视图会被重新创建,处理不好可能产生泄露。



 
Android技术之家 更多文章 自动展开标题通知栏,兼容各个版本,欢饮大家关注,并参与讨论。 minSdkVersion、targetSdkVersion、targetApiLevel的区别 ndk调用实战 GreenDao数据库操作 大家来找茬 看看下面这个单例模式有什么不妥的
猜您喜欢 成为大牛Java程序员一定要看的10本书 大话Android Drawable类型<中> 饿了么移动APP的架构演进 【TGC现场】阮佳:置身于西方但我还回归 再议携程Android动态加载框架DynamicAPK