Pet*_*rev 7 android kotlin kotlin-native kotlin-multiplatform
我目前正在开发一个简单的 KMM 模块,需要它Context才能执行一些操作。我知道通过扩展Application类和进行依赖注入来实现这一点的方法。我现在正在尝试做什么 - 使该模块开箱即用,无需修改manifest或在启动时进行手动注入。我只是想知道这样做是不是一种不好的做法:
@SuppressLint("StaticFieldLeak")
object SomeUtil {
private val context = Activity().applicationContext
}
Run Code Online (Sandbox Code Playgroud)
由于applicationContext返回Context整个应用程序并且我们正在初始化它一次,会出现泄漏吗?还是还有其他几点没做到?
也许还有其他一些可能性可以从模块获取应用程序上下文?我已经看到了一些从线程中检索它的示例,但据我了解,这将被(或已经)弃用。
UPD:这会导致错误。Activity()似乎是null。那么,有什么想法如何在没有 DI 和“MyApplication”的情况下实现这一目标吗?
这是 Android 库中的一个常见问题 - 如何在无法访问应用程序代码库的情况下获取应用程序上下文?这就是为什么你经常用类似的东西来初始化SharedPrefHelper.init(applicationContext)库Application.onCreate()
由于 KMM 共享代码是一个库,因此您会遇到类似的问题。Android 应用程序启动是一个 androidx lib,旨在解决这个问题(以及提高启动性能)。
粗略示例(共享代码中的所有内容):
// In androidMain
class MySqlDelightInitialiser : Initializer<SqlDriver> {
override fun create(context: Context): SqlDriver {
val driver = createDriver(context)
MyLibraryObject.init(context, driver)
return driver
}
override fun dependencies(): List<Class<out Initializer<*>>> {
return emptyList()
}
}
// In androidMain/AndroidManifest
<application>
<provider
android:name="androidx.startup.InitializationProvider"
android:authorities="${applicationId}.sql-delight-initialiser"
android:exported="false"
tools:node="merge"
tools:replace="android:authorities"
>
<meta-data
android:name="my.package.SqlDelightInitialiser"
android:value="androidx.startup"
/>
</provider>
</application>
Run Code Online (Sandbox Code Playgroud)
好吧,我首先要说的是,这并不是一个真正的 KMM 问题。这仅适用于 Android 代码。
据我所知,不,如果没有一些半黑客解决方案,就无法静态地、全局地访问应用程序上下文。这是一个长期存在的问题,目前还没有一个好的解决方案。
Crashlytics 通过注册 ContentProvider 做了一些奇怪的事情,ContentProvider 的唯一目的是获取应用程序并使其可用。假设您以 aar 形式发布,它会为您注册 ContentProvider。
https://firebase.googleblog.com/2016/12/how-does-firebase-initialize-on-android.html
我不建议这样做。我非常喜欢自己配置库上下文初始化,但您可以尝试 ContentProvider 路线。
| 归档时间: |
|
| 查看次数: |
4088 次 |
| 最近记录: |