- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我在尝试找出实现 UseCase 的最佳方法时遇到了问题。我看过这样的例子:
我不喜欢的是:
我认为是对上述内容的升级:
Android 架构蓝图
他们正在使用协程,现在结果在 ViewModel 中,而且好多了。但同样:没有进展
我的问题是每个人都在使用 RxJava,因为其他人都在使用它。我看到很多人认为这是“在后台执行任务”的东西。但这对我来说太多了。我不需要它。
我看过一些使用协程 channel 的例子,但它们真的很丑。
最近我偶然发现了 Roman Elizarov 的这篇文章:
https://medium.com/@elizarov/callbacks-and-kotlin-flows-2b53aa2525cf
我做了这样的事情:
class ProduceValueWithProgressUseCase @Inject constructor(
private val executor: Executor
) {
operator fun invoke(): Flow<Result<Int, Int>> {
return callbackFlow {
val callback = object : CallbackProgress {
override fun onProgress(result: Int) {
offer(Result.Loading(result))
}
override fun onSuccess(result: Int) {
offer(Result.Success(result))
close()
}
override fun onError(e: Exception) {
offer(Result.Error(e))
close()
}
}
val producer = ValueWithProgressProducer(callback)
executor.execute(producer)
awaitClose {}
}
}
}
这个想法是“生产者”这个东西像大多数“旧”API 一样使用回调来传播数据。所以我想通过流将代码传播到 ViewModel,而不是将回调放在那里。像这样:
viewModelScope.launch {
produceValueWithProgressUseCase().collect {
when (it) {
is Success -> view.showWithProgressResult(it.data)
is Loading -> view.updateProgress(it.progress)
else -> view.showError()
}
}
}
是的,基本上 Flows API 会为我完成这项工作。我什至创建了一个小应用程序来测试它,我只生成数字,它运行良好。我不喜欢的是:
太多的 ExperimentalCoroutinesApi 注释必须到位。例如这里(抱歉格式化):
@Suppress("NOTHING_TO_INLINE")@ExperimentalCoroutinesApipublic inline fun callbackFlow(@BuilderInference noinline block :暂停ProducerScope.() -> Unit): Flow = channel 流( block )
以上代码是文件的一部分:kotlinx.coroutines.flow.Builders.kt 来自版本:kotlinx-coroutines-core-1.3.2
所以最后我看到了一些看起来像是在为明年的改变而尖叫的东西。因此,请发表您的看法。
最佳答案
因此,我将根据您在问题描述中提出的内容向您展示一个实现,请随时提出更多问题,甚至提出改进建议。
当前的实现基于 Kotlin Coroutines 1.3.2 的稳定版本。
回调接口(interface)
interface CallbackProgress {
suspend fun onProgress(result: Int)
suspend fun onSuccess(result: Int)
suspend fun onError(exception: Exception)
}
制作人迭代并执行一些回调的方法。试图模仿你的。
class Producer(private val callback: CallbackProgress) {
suspend fun execute(fail: Boolean) {
(0 until 10).forEach {
when {
it < 9 -> callback.onProgress(it)
fail -> callback.onError(InterruptedException("blabla"))
else -> callback.onSuccess(it)
}
delay(500)
}
}
}
状态
sealed class State {
class Success(val value: Int) : State()
class Loading(val progress: Int) : State()
class Error(val exception: Exception) : State()
}
互动者您在这里需要小心,因为排放应该在同一个协同程序中完成,否则您将需要使用不稳定的 API,例如 channelFlow .
class UseCase {
operator fun invoke(fail: Boolean) = flow {
val callback = object : CallbackProgress {
override suspend fun onSuccess(result: Int) {
withContext(coroutineContext) { emit(State.Success(result)) }
}
override suspend fun onError(exception: Exception) {
withContext(coroutineContext) { emit(State.Error(exception)) }
}
override suspend fun onProgress(result: Int) {
withContext(coroutineContext) { emit(State.Loading(result)) }
}
}
Producer(callback).execute(fail)
}
}
为了测试以上内容,我编写了以下内容,其中演示了故障排放和非故障排放。
fun main() = runBlocking {
val useCase = UseCase()
useCase(true).collect {
when (it) {
is State.Loading -> println("State for failure [Loading -> ${it.progress}]")
is State.Success -> println("State for failure [Success -> ${it.value}]")
is State.Error -> println("State for failure [Error -> ${it.exception.message}]")
}
}
useCase(false).collect {
when (it) {
is State.Loading -> println("State without failure [Loading -> ${it.progress}]")
is State.Success -> println("State without failure [Success -> ${it.value}]")
is State.Error -> println("State without failure [Error -> ${it.exception.message}]")
}
}
}
输出
State for failure [Loading -> 1]
State for failure [Loading -> 2]
State for failure [Loading -> 3]
State for failure [Loading -> 4]
State for failure [Loading -> 5]
State for failure [Loading -> 6]
State for failure [Loading -> 7]
State for failure [Loading -> 8]
State for failure [Error -> blabla]
-------------------------------------
State without failure [Loading -> 0]
State without failure [Loading -> 1]
State without failure [Loading -> 2]
State without failure [Loading -> 3]
State without failure [Loading -> 4]
State without failure [Loading -> 5]
State without failure [Loading -> 6]
State without failure [Loading -> 7]
State without failure [Loading -> 8]
State without failure [Success -> 9]
关于android - 有进展的整洁架构用例[无 RxJava],我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58714389/
是否可以简化在裸机上运行的这条链: 具有随时间变化的副本数的 StatefulSet 服务 使用 proxy-next-upstream: "error http_502 timeout invali
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
我需要为应用程序制定架构。它专为销售产品而设计。 系统每天将接受大约 30-40k 的新产品。它将导致在表 product 中创建新记录。 系统应保留价格历史记录。用户应该能够看到产品 A 的价格在去
我需要一些帮助来理解 PHP 的内部工作原理。 还记得,在过去,我们曾经写过 TSR(Terminate and stay resident)例程(pre-windows 时代)吗?一旦该程序被执行,
1.Nginx 基础架构 nginx 启动后以 daemon 形式在后台运行,后台进程包含一个 master 进程和多个 worker 进程。如下图所示: master与
本文深入探讨了Kubernetes(K8s)的关键方面,包括其架构、容器编排、网络与存储管理、安全与合规、高可用性、灾难恢复以及监控与日志系统。 关注【TechLeadCloud】,
我知道 CNN 的工作原理,包括每一层的用途(Dropout、Pooling 等)。但是,在为新数据集设计 CNN 时,我不知道要使用多少个 Conv-Relu-Pool 层,在最终获得输出之前我应该
在基于 REST 的架构中,资源和方法之间有什么区别。有吗? 最佳答案 资源是您的应用程序定义的东西;它们与物体非常相似。方法是 HTTP 动词之一,例如 GET、POST、PUT、DELETE。它们
我想用 oneOf仅在 xyType 的值上不同的模式属性(property)。我想要其中两个:一个是 xyType设置为 "1"第二个在哪里xyType是 任何其他值 .这可以使用 json 模式完
寻求 PHP 架构师的建议! 我对 PHP 不是很熟悉,但已经接管了一个用该语言编写的大型分析包的维护工作。该架构旨在将报告的数据读取到大型键/值数组中,这些数组通过各种解析模块传递,以提取每个模块已
这些存在吗? 多年来,我一直是大型强类型面向对象语言(Java 和 C#)的奴隶,并且是 Martin Fowler 及其同类的信徒。 Javascript,由于它的松散类型和函数性质,似乎不适合我习
我已经阅读了 Manning 的 Big Data Lambda Architecture ( http://www.manning.com/marz/BD_meap_ch01.pdf ),但仍然无法
在过去的几年里,我做了相当多的 iOS 开发,所以我非常熟悉 iOS 架构和应用程序设计(一切都是一个 ViewController,您可以将其推送、弹出或粘贴到选项卡栏中)。我最近开始探索正确的 M
我有以下应用程序,我在其中循环一些数据并显示它。 {{thing.title}} {{thing.description}}
昨天我和我的伙伴讨论了我正在开发的这个电子购物网站的架构。请注意,我为此使用 ASP.NET。他非常惊讶地发现我没有将添加到购物车的项目保留在 ArrayList 或其他通用列表中,而是使用 LINQ
我正在使用在 tridion 蓝图层次结构中处于较低位置的出版物。从蓝图中较高级别的出版物继承的一些内容和模式不适合我的出版物,并且永远不会被我的出版物使用。 我将跟进添加这些项目的内部团队,并尝试说
我目前已经在 Cassandra 中设计了一个架构,但我想知道是否有更好的方法来做事情。基本上,问题在于大多数(如果不是全部)读取都是动态的。我构建了一个分段系统作为应用程序服务,读取动态自定义查询(
我正在按照 documentation 中给出的 icingaweb UI v 2.0 布局执行在服务器上设置 icinga 的步骤。 。我成功进入设置页面,该页面要求您输入 token ,然后按照步
我必须保存来自不同社交媒体的用户的不同个人资料。例如用户可能有 1 个 Facebook 和 2 个 Twitter 个人资料。如果我保存每个配置文件它作为新文档插入不同的集合中,例如 faceboo
我的团队使用 Puppet 架构,该架构目前可在多个环境(流浪者、暂存、生产)中容纳单个应用程序。 我们现在想要扩展此设置的范围以支持其他应用程序。他们中的许多人将使用我们已经定义的现有模块的子集,而
我是一名优秀的程序员,十分优秀!