为什么 JakeWharton 建议 App 只要用到一个 Activity?

安卓开发大神级人物 JakeWharton 前不久在接受采访时提出一个颇具争议而又没有给出原因的建议:一个 App 只需要一个 Activity ,你可以使用 Fragments,只是别用 Fragments 回退栈。

针对这一言论,有关 JakeWharton 建议的背后原因的一个提问迅速在 Reddit 国外网站的安卓开发频道引发热评。

众多安卓开发人员纷纷提出自己的见解。其中获赞最高的一条,甚至得到 JakeWharton 本人的亲自赞评。

我们不妨来看看这条回答具体都提到哪些内容,对 Activity 和 Fragment 之间的爱恨情仇又有何独到的见解,凭什么能得到 JakeWharton 本尊的青睐有加。


因为 Activity 是一个程序入口。你可以将其视为 app 的一个 main 函数。站在用户的立场上,通常你进入 app 的方式可能包括以下几种:

  • launcher 桌面程序(main 函数入口);

  • 来自参数化 main 函数入口的通知栏,并且导航到 app 的指定位置;

  • 如果你做的是一个相机应用,那么需要处理图片请求的 intents;

  • 如果你做的是一个社交产品,那么需要处理 share 请求的 intents;

差不多类似这些场景。

但是,如果你真的不用分享和来自应用的 intents 的话,并且唯一的程序入口就是 launcher 桌面,别为每一个页面创建一个新的入口。这样做其实没有意义。为什么没有意义?因为这种场景下,进程死掉后 launcher 能够启动任何你应用中的 Activity 页面。

Fragments 是处理生命周期事件的视图控制器,并且非常不错。然而,Fragments 回退栈简直垃圾;回退栈变化监听器总是不正常地被调用( 1 次 transaction 三次调用?),并且不告诉你调用什么,而在恢复事务时也不知道哪些 fragments 是可用的。

你可以给事务添加 tag 标签,然后从栈中弹出操作,但是仅仅是一个 main -> Events -> Details(id=123) 的操作流程就相当繁琐了。

同样的,一旦你将一个 Fragment 放进回退栈中,我个人不知道它的生命周期开始做什么。我曾经遇到过一个后台中的 fragment 被调用四次 onCreateView() 方法,我甚至不知道究竟怎么了。而没有位于回退栈中的 Fragments 是可以被预见的。它们的动画支持有点古怪,但至少它们还能使用。

所以如果你想知道哪些 Fragments 是你能够操作的并且哪些 views 是你正在展示的并且能够在你自己的导航状态控制之中,那么你应该自己处理导航操作。把“应用逻辑”抽象化到一个 presenter(亦枫注:MVP 模式)中听起来来很棒,但是你是不是脱离了应用视图层里面的真实情况?

但是单一 activity 的优势是什么?

更简单的生命周期处理(例如,当 app 进入后台时,你只需要处理 onStop 方法),更少错误空间,和更多控制。同样的,你可以移动视图层外面的导航状态到 domain 层,或者至少到 presenter 中。不需要太多 view.navigateToDetail(songId) 之类的东西,你只需要在你的 presenter 或者 ViewModel 或者无论哪些时髦的用法中使用 backstack.goTo(SongKey.create(songId)) 就行。借助一个合适的库,当你到了 onResume 时它会自动将这些导航调用加入队列,并且不会致使 fragment 事务发生崩溃,非常得好。

尽管 Google 给出的案例也在用 commitAllowingStateLoss(),我有使用commitNow() 的动画爱好。在我看来,单个 activity 能够看得见的好处就是,页面间共享 views 的能力,取代通过使用 <include 标签在 18 个布局文件重复视图。其他当然是更简单的导航操作。


以上便是深得 JakeWharton 大神心意的一条回答。话虽如此,但是系统 Fragment 存在的未解之谜或者说出乎你意料的坑实在太多。如果一定要在多 activity 部分 fragments 和单 activity 多 fragments 之间选择的话,我想不只是我,很多人还是毫不犹豫地选择前者。

更多讨论内容,参见:

https://www.reddit.com/r/androiddev/comments/7hq00q/why_does_jake_wharton_recommend_one_activity_for/

备注:关于使用 Fragments 的那些道道,我之前也写过相关总结性的文章来,感兴趣地不妨去我的博客搜索看看。

赞助商

优秀人才不缺工作机会,只缺适合自己的好机会。但是他们往往没有精力从海量机会中找到最适合的那个。

100offer 会对平台上的人才和企业进行严格筛选,让「最好的人才」和「最好的公司」相遇。

扫描下方二维码,注册 100offer,谈谈你对下一份工作的期待。一周内,收到 5-10 个满足你要求的好机会!

热门分享

重大改革!Python,最接近人工智能的语言,将被加入高考科目!

如何判断一个 GitHub 开源项目是否值得一用?

在Android Studio中注册一个新的Activity需要几个步骤: 1. 创建新的Activity类:首先,你需要创建一个新的Java或Kotlin类,让它继承自`Activity`或其子类(如`AppCompatActivity`,如果你使用的是AndroidX库)。通常,我们会通过Android Studio提供的模板快速创建一个Activity。 2. 在AndroidManifest.xml中注册:在你的项目中的`AndroidManifest.xml`文件中,你需要添加对应的`<activity>`标签以声明并注册你新创建的Activity。这是必须的步骤,因为Android系统需要通过这个文件知道你的应用中有哪些可访问的Activity。以下是一个注册Activity的例子: ```xml <application android:allowBackup="true" android:icon="@mipmap/ic_launcher" android:label="@string/app_name" android:roundIcon="@mipmap/ic_launcher_round" android:supportsRtl="true" android:theme="@style/AppTheme"> <activity android:name=".MainActivity"> <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> </activity> <!-- 注册新的Activity --> <activity android:name=".NewActivity" /> </application> ``` 在上面的XML代码中,`.MainActivity`是已经注册的Activity,而`.NewActivity`是你新创建的Activity的名称。注意,你需要将`.NewActivity`替换为你自己的Activity类名。 3. 配置Intent Filter(可选):如果你希望这个新的Activity能够作为启动Activity或者响应特定类型的Intent,则需要配置相应的`<intent-filter>`。例如,如果你想让你的Activity能够作为应用的入口点,那么你需要配置如`MAIN`和`LAUNCHER`这样的action和category,如上面的`MainActivity`所示。 完成以上步骤后,你的新Activity就注册完毕了,可以在应用中被调用和使用。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值