Android设计模式之InitContentProvider

作者: hglfNg | 来源:发表于2019-06-27 16:49 被阅读17次

本文要讲的设计模式不是常常听到的GoF23种设计模式,也不会讨论java中茴香豆的茴,哦不,单例有多少种写法。本文主要介绍在Android开发中一些鲜为人知,但实际上广泛使用的设计模式。

所谓设计模式就是介于基本代码和架构之间的一种方法,用于解决特定的问题。在开发者之间已经达成某种默契,不需要特别说明,你在阅读这些代码的时候就能理解作者的意图。

问题

当一个app要在启动的时候做一些初始化操作,通常我们会重写Appliction的onCreate,例如这样:

class App : Application() {
    override fun attachBaseContext(base: Context?) {
        Log.d("App", "attachBaseContext: ${base?.javaClass?.canonicalName}")
        super.attachBaseContext(base)
    }

    override fun onCreate() {
        Log.d("App", "onCreate")
        super.onCreate()
        //init 操作
    }
}

如果是我们自己写app这样做没有问题,但是如果是写一个sdk,或者一个开源库,就必须让调用者在onCreate调用某个方法,这样就不太友好,有没有什么方法让调用者只需要在gradle配置中引入我们的库就能在自动在app启动的时候自动初始化呢?答案就是InitContentProvider

InitContentProvider

作为四大组件之一的ContentProvider,顾名思义被设计出来用于跨应用间的内容分享。但是由于它的一个独特特性:在应用被打开时每个注册的ContnentProvider都会被实例化并调用onCreate方法。
例如我们创建一个ContnentProvider 类

class ShadowProvider : ContentProvider() {

    override fun onCreate(): Boolean {
        Log.d("ShadowProvider", "onCreate with context:${context.javaClass.canonicalName}")
        return false
    }

    override fun insert(uri: Uri, values: ContentValues?): Uri? = null

    override fun query(
        uri: Uri,
        projection: Array<String>?,
        selection: String?,
        selectionArgs: Array<String>?,
        sortOrder: String?
    ): Cursor? = null
    
    override fun update(uri: Uri, values: ContentValues?, selection: String?, selectionArgs: Array<String>?): Int  = 0
    override fun delete(uri: Uri, selection: String?, selectionArgs: Array<String>?): Int = 0
    override fun getType(uri: Uri): String?  = null

}

并且在manifest中注册这个contentProvider,在app启动的时候就能看到如下日志输出

2019-06-27 16:03:44.164 31473-31473/? D/App: attachBaseContext: android.app.ContextImpl
2019-06-27 16:03:44.167 31473-31473/? D/ShadowProvider: onCreate with context:github.hotstu.shadowprovider.demo.App
2019-06-27 16:03:44.171 31473-31473/? D/App: onCreate

可以看到contentProvder的onCreate方法调用是在Application的attachBaseContext和onCreate之间

哪些库用了这个技术

firebase、androidx.camerax、androidx.worker、 crashlytics
等等,可以看到第一方库都在用这个模式。在使用者无知觉的情况下完成初始化.

性能影响

虽然google官方告诉大家开发者不要使用反射,但实际上可以看到framework内部的反射用得飞起,这个cotentprovder的初始化也不例外的使用了反射,所以为了影响app的启动速度,请不要滥用

如何禁用

第三方的库在manifest中搞了这个,如何禁用?

        <provider android:name="com.crashlytics.android.CrashlyticsInitProvider"
                  tools:node="remove"/>
        <provider android:name="com.google.firebase.provider.FirebaseInitProvider"
                  tools:node="remove"/>

只想在特定版本下启用?

例如camera2相关api只在api21后才支持,但app需要兼容之前的版本, 解决办法:在value和value-21中声明bool值, 在 android:enabled中引用,这样在5.0之前改组件就被禁用了

            android:name="androidx.camera.camera2.impl.Camera2Initializer"
            android:authorities="${applicationId}.camerax-init"
            android:enabled="@bool/camerax_enable"
            android:exported="false"
            android:initOrder="100"
            android:multiprocess="true" />

多进程

默认情况下contentProvider只会创建一个,如果app中用到了多进程,需要添加android:multiprocess="true"声明,这样每个进程中都会创建一次

相关文章

网友评论

    本文标题:Android设计模式之InitContentProvider

    本文链接:https://www.haomeiwen.com/subject/fjmtcctx.html