作者热门文章
- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我一直在尝试使用 kotlin 多平台,它非常棒,但线程让我难堪。线程之间的状态卡住在概念上是有意义的,并且在来回传递小对象或基元的简单示例中工作正常,但在现实世界的应用程序中,我无法绕过 InvalidMutabilityException。
从 Android 应用程序中获取以下常见代码片段
class MainViewModel(
private val objectWhichContainsNetworking: ObjectWhichContainsNetworking
)
private var coreroutineSupervisor = SupervisorJob()
private var coroutineScope: CoroutineScope = CoroutineScope(Dispatchers.Main + coreroutineSupervisor)
private fun loadResults() {
// Here: Show loading
coroutineScope.launch {
try {
val result = withContext(Dispatchers.Default) { objectWhichContainsNetworking.fetchData() }
// Here: Hide loading and show results
} catch (e: Exception) {
// Here: Hide loading and show error
}
}
}
没有什么非常复杂的,但如果在通用代码中使用并从 Kotlin/Native 运行,那么在 MainViewModel 上 pow InvalidMutabilityException。
最佳答案
在您的原始代码中,您引用了父对象的一个字段,这会导致您捕获整个父对象并将其卡住。这不是协程的问题。协程遵循与 Kotlin/Native 中所有其他并发库相同的规则。当您跨线程时,它会卡住 lambda。
class MainViewModel(
private val objectWhichContainsNetworking: ObjectWhichContainsNetworking
)
//yada yada
private fun loadResults() {
coroutineScope.launch {
try {
val result = withContext(Dispatchers.Default) {
//The reference to objectWhichContainsNetworking is a field ref and captures the whole view model
objectWhichContainsNetworking.fetchData()
}
} catch (e: Exception) {}
}
}
为了防止这种情况发生:
class MainViewModel(
private val objectWhichContainsNetworking: ObjectWhichContainsNetworking
){
init{
ensureNeverFrozen()
}
//Etc
内存模型最复杂的就是这个。习惯被捕获的东西并避免它。当你习惯了它并不难,但你需要学习基础知识。
关于使用协程的 Kotlin/Native 多线程,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/63485259/
在我的设置中,我试图有一个界面 Table继承自 Map (因为它主要用作 map 的包装器)。两个类继承自 Table - 本地和全局。全局的将有一个可变的映射,而本地的将有一个只有本地条目的映射。
Rust Nomicon 有 an entire section on variance除了关于 Box 的这一小节,我或多或少地理解了这一点和 Vec在 T 上(共同)变体. Box and Vec
我是一名优秀的程序员,十分优秀!