Android开发技术之InitContentProvider

前言

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"声明,这样每个进程中都会创建一次

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 2.1 Activity 2.1.1 Activity的生命周期全面分析 典型情况下的生命周期:在用户参与的情况下...
    AndroidMaster阅读 3,280评论 0 8
  • 本文出自 Eddy Wiki ,转载请注明出处:http://eddy.wiki/interview-androi...
    eddy_wiki阅读 3,379评论 0 20
  • 一、区别 nvm:nodejs 版本管理工具。 也就是说:一个 nvm 可以管理很多 node 版本和 npm 版...
    Aniugel阅读 8,277评论 0 7
  • AVL树:平衡的二叉查找树 AVL树是带有平衡条件的二叉查找树,一般是用平衡因子差值判断是否平衡并通过旋转来实现平...
    liuzx32阅读 362评论 0 0
  • 今天我觉得妈妈有点讨厌,什么都要逆着我的意,不过今天也是最累的一天。 今天妈妈和我说好了,今天赶集要带...
    生机勃勃888阅读 317评论 2 4

友情链接更多精彩内容