- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
我最近决定仔细研究 Google 发布的新 Android 架构组件,尤其是使用他们的 ViewModel 生命周期感知类到 MVVM 架构和 LiveData。
只要我处理单个 Activity 或单个 Fragment,一切都很好。
但是,我找不到处理 Activity 切换的好解决方案。比如说,为了一个简短的例子,Activity A 有一个启动 Activity B 的按钮。
startActivity() 会在哪里处理?
按照 MVVM 模式,clickListener 的逻辑应该在 ViewModel 中。但是,我们希望避免在其中引用 Activity。因此,将上下文传递给 ViewModel 不是一种选择。
我缩小了几个看起来“可以”的选项,但找不到“这里是如何做”的任何正确答案。
选项 1:在 ViewModel 中有一个枚举,其值映射到可能的路由(ACTIVITY_B、ACTIVITY_C)。将此与 LiveData 结合使用。Activity 会观察这个 LiveData,当 ViewModel 决定应该启动 ACTIVITY_C 时,它只是 postValue(ACTIVITY_C)。然后Activity就可以正常调用startActivity()了。
选项 2 :常规界面模式。原理与选项 1 相同,但 Activity 将实现接口(interface)。不过,我觉得与此有更多的耦合。
选项 3 :消息选项,例如 Otto 或类似选项。 ViewModel 发送一个广播,Activity 接收它并启动它必须要做的事情。此解决方案的唯一问题是,默认情况下,您应该将该广播的注册/注销放在 ViewModel 中。所以没有帮助。
选项 4:在某个地方拥有一个大的路由类,如单例或类似的,可以调用它来将相关路由分派(dispatch)给任何 Activity 。最终通过接口(interface)?所以每个 Activity (或 BaseActivity)都会实现
IRouting { void requestLaunchActivity(ACTIVITY_B); }
当你的应用开始有很多 fragment/Activity 时,这个方法让我有点担心(因为路由类会变得非常庞大)
就是这样。那是我的问题。你们怎么处理这个?你会选择我没有想到的选择吗?您认为哪个选项最相关,为什么?推荐的 Google 方法是什么?
PS : 没有把我带到任何地方的链接1 - Android ViewModel call Activity methods2 - How to start an activity from a plain non-activity java class?
最佳答案
NSimon,你开始使用 AAC 真是太好了。
我写了一个 issue在aac's-github之前关于那个。
有几种方法可以做到这一点。
一种解决方案是使用
WeakReference到保存 Activity 上下文的 NavigationController。这是在 ViewModel 中处理上下文绑定(bind)内容的常用模式。
出于几个原因,我强烈拒绝这样做。首先:这通常意味着您必须保留对修复上下文泄漏的 NavigationController 的引用,但根本不能解决架构问题。
最好的方法(在我看来)是使用 LiveData,它具有生命周期意识,可以做所有想要的事情。
例子:
class YourVm : ViewModel() {
val uiEventLiveData = SingleLiveData<Pair<YourModel, Int>>()
fun onClick(item: YourModel) {
uiEventLiveData.value = item to 3 // can be predefined values
}
}
之后,您可以在 View 中监听变化。
class YourFragmentOrActivity {
//assign your vm whatever
override fun onActivityCreated(savedInstanceState: Bundle?) {
var context = this
yourVm.uiEventLiveData.observe(this, Observer {
when (it?.second) {
1 -> { context.startActivity( ... ) }
2 -> { .. }
}
})
}
}
请注意我使用了修改后的 MutableLiveData,否则它总是会为新的观察者发出最新的结果,这会导致不良行为。例如,如果您更改 Activity 并返回,它将以循环结束。
class SingleLiveData<T> : MutableLiveData<T>() {
private val mPending = AtomicBoolean(false)
@MainThread
override fun observe(owner: LifecycleOwner, observer: Observer<T>) {
if (hasActiveObservers()) {
Log.w(TAG, "Multiple observers registered but only one will be notified of changes.")
}
// Observe the internal MutableLiveData
super.observe(owner, Observer { t ->
if (mPending.compareAndSet(true, false)) {
observer.onChanged(t)
}
})
}
@MainThread
override fun setValue(t: T?) {
mPending.set(true)
super.setValue(t)
}
/**
* Used for cases where T is Void, to make calls cleaner.
*/
@MainThread
fun call() {
value = null
}
companion object {
private val TAG = "SingleLiveData"
}
}
为什么这种尝试比使用弱引用、接口(interface)或任何其他解决方案更好?
因为此事件将 UI 逻辑与业务逻辑分开。它也可以有多个观察者。它关心生命周期。它不会泄漏任何东西。
您也可以通过使用 RxJava 而不是 LiveData 通过使用 PublishSubject 来解决它。 (addTo
需要 RxKotlin )
注意不要通过在 onStop() 中释放订阅来泄露订阅。
class YourVm : ViewModel() {
var subject : PublishSubject<YourItem> = PublishSubject.create();
}
class YourFragmentOrActivityOrWhatever {
var composite = CompositeDisposable()
onStart() {
YourVm.subject
.subscribe( { Log.d("...", "Event emitted $it") }, { error("Error occured $it") })
.addTo(compositeDisposable)
}
onStop() {
compositeDisposable.clear()
}
}
还要注意 ViewModel 绑定(bind)到 Activity 或 Fragment。您不能在多个 Activity 之间共享 ViewModel,因为这会破坏“Livecycle-Awareness”。
如果您需要使用 room 之类的数据库来保存数据或使用包裹共享数据。
关于android - MVVM 模式和 startActivity,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46727276/
我最近在/ drawable中添加了一些.gifs,以便可以将它们与按钮一起使用。这个工作正常(没有错误)。现在,当我重建/运行我的应用程序时,出现以下错误: Error: Gradle: Execu
Android 中有返回内部存储数据路径的方法吗? 我有 2 部 Android 智能手机(Samsung s2 和 s7 edge),我在其中安装了一个应用程序。我想使用位于这条路径中的 sqlit
这个问题在这里已经有了答案: What's the difference between "?android:" and "@android:" in an android layout xml f
我只想知道 android 开发手机、android 普通手机和 android root 手机之间的实际区别。 我们不能从实体店或除 android marketplace 以外的其他地方购买开发手
自Gradle更新以来,我正在努力使这个项目达到标准。这是一个团队项目,它使用的是android-apt插件。我已经进行了必要的语法更改(编译->实现和apt->注释处理器),但是编译器仍在告诉我存在
我是android和kotlin的新手,所以请原谅要解决的一个非常简单的问题! 我已经使用导航体系结构组件创建了一个基本应用程序,使用了底部的导航栏和三个导航选项。每个导航选项都指向一个专用片段,该片
我目前正在使用 Facebook official SDK for Android . 我现在正在使用高级示例应用程序,但我不知道如何让它获取应用程序墙/流/状态而不是登录的用户。 这可能吗?在那种情
我在下载文件时遇到问题, 我可以在模拟器中下载文件,但无法在手机上使用。我已经定义了上网和写入 SD 卡的权限。 我在服务器上有一个 doc 文件,如果用户单击下载。它下载文件。这在模拟器中工作正常但
这个问题在这里已经有了答案: What is the difference between gravity and layout_gravity in Android? (22 个答案) 关闭 9
任何人都可以告诉我什么是 android 缓存和应用程序缓存,因为当我们谈论缓存清理应用程序时,它的作用是,缓存清理概念是清理应用程序缓存还是像内存管理一样主存储、RAM、缓存是不同的并且据我所知,缓
假设应用程序 Foo 和 Eggs 在同一台 Android 设备上。任一应用程序都可以获取设备上所有应用程序的列表。一个应用程序是否有可能知道另一个应用程序是否已经运行以及运行了多长时间? 最佳答案
我有点困惑,我只看到了从 android 到 pc 或者从 android 到 pc 的例子。我需要制作一个从两部手机 (android) 连接的 android 应用程序进行视频聊天。我在想,我知道
用于使用 Android 以编程方式锁定屏幕。我从 Stackoverflow 之前关于此的问题中得到了一些好主意,并且我做得很好,但是当我运行该代码时,没有异常和错误。而且,屏幕没有锁定。请在这段代
文档说: android:layout_alignParentStart If true, makes the start edge of this view match the start edge
我不知道这两个属性和高度之间的区别。 以一个TextView为例,如果我将它的layout_width设置为wrap_content,并将它的width设置为50 dip,会发生什么情况? 最佳答案
这两个属性有什么关系?如果我有 android:noHistory="true",那么有 android:finishOnTaskLaunch="true" 有什么意义吗? 最佳答案 假设您的应用中有
我是新手,正在尝试理解以下 XML 代码: 查看 developer.android.com 上的文档,它说“starStyle”是 R.attr 中的常量, public static final
在下面的代码中,为什么当我设置时单选按钮的外观会发生变化 android:layout_width="fill_parent" 和 android:width="fill_parent" 我说的是
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 9
假设我有一个函数 fun myFunction(name:String, email:String){},当我调用这个函数时 myFunction('Ali', 'ali@test.com ') 如何
我是一名优秀的程序员,十分优秀!