百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程字典 > 正文

使用 ContentProvider无侵入获取Context_无租使用房产如何缴纳房产税

toyiye 2024-04-03 23:10 21 浏览 0 评论

前言

  • 在Android中,使用三方库或二方库时,通常都需要使用Context进行初始化
  • 这篇文章将介绍一种基于ContentProvider 机制实现的无侵入获取Context的方法,希望能帮上忙

目录


1. 获取 Context 的常规方法

我们知道Context本身是一个抽象类,所以我们获取Context实际上是获取Context的实现类。简单来说:Context 使用了装饰模式,除了 ContextImpl 外,其他 Context 都是 ContextWrapper 的子类,具体分为:Application、Activity、Service;

1.1 获取 Application 对象

既然Application是Context的实现类,那么我们就可以直接使用Application对象来初始化第三方库,同时也可以使用一个静态方法将对象暴露出去:

// MainApplication.kt

class MainApplication : Application() {

    companion object {
        lateinit var application: Application
            get
    }

    override fun onCreate() {
        super.onCreate()
        application = this

        // 初始化第三方库
    }
}
复制代码

1.2 获取 Activity & Service 对象

同样地, Activity & Service也是Context的实现类,那么我们就可以在程序运行过程中,按需初始化第三方库。例如使用Glide时,并不需要一开始就调用Glide#with(Context),只需要在显示图片的时候调用即可;

1.3 小结

  • 优点

最常用的方式,实现简单,没有性能 / 稳定性风险;可以按需初始化第三方库 & 懒加载

  • 缺点

需要获取ApplicationContext / Context(依赖方与库代码强耦合),不利于组件化

下面,我将介绍两种无侵入获取Context的方法,将涉及到Android进程的启动流程,若还不了解,请务必阅读文章:《Android | 带你理解 Application 的创建过程》


2. 反射 ActivityThread 获得 ApplicationContext(不推荐)

这一节介绍一种通过ActivityThread.java获得Application的方法,具体如下:

2.1 源码分析

我们都知道,在启动四大组件(Activity、Service、ContentProvider, BroadcastReceiver)时,如果对应的进程未启动,就需要先创建进程,相应地也会创建一个Application对象,即:

  • 在system_server进程,通过AMS#getProcessRecordLocked(...)获取进程信息(ProcessRecord);
  • 若不存在,则调用AMS#startProcessLocked(...)创建进程;
  • 在Zygote孵化目标进程之后,在目标进程反射执行ActivityThread#main(),并最终在ActivityThread#handleBindApplication(...)中创建Application对象。

ActivityThread.java

Application mInitialApplication;

public Application getApplication() {
        return mInitialApplication;
}
    
private void handleBindApplication(AppBindData data) {
    // ...
    Application app;
    // data.info 为 LoadedApk.java
    app = data.info.makeApplication(data.restrictedBackupMode, null);
    // ...
    mInitialApplication = app;
    // ...
}
复制代码

可以看到,创建Application对象之后会保存在mInitialApplication属性中,那么如果我们可以访问到这个属性,是不是就可以获得Application对象了呢?

首先,我们需要获得ActivityThread对象,那么我们先在源码中寻找创建ActivityThread对象的地方:

ActivityThread.java

private static volatile ActivityThread sCurrentActivityThread;

public static ActivityThread currentActivityThread() {
    return sCurrentActivityThread;
}
    
// (简化)
public static void main(String[] args) {
    Looper.prepareMainLooper();
    
    // 创建 ActivityThread 对象
    ActivityThread thread = new ActivityThread();
    thread.attach(false, startSeq);
    
    Looper.loop();
}

private void attach(boolean system, long startSeq) {
    sCurrentActivityThread = this;
}
复制代码

可以看到,ActivityThread对象存储在静态变量sCurrentActivityThread中,那么我们就可以写反射代码了。

2.2 使用步骤

// 新建文件 Context.kt

private var application: Context? = null

fun context(): Context {
    if (null == application) {
        try {
            val activityThread: Any
            val clazz = Class.forName("android.app.ActivityThread")
            val currentActivityThread = clazz.getMethod("currentActivityThread").apply {
                isAccessible = true
            }
            val getApplication = clazz.getMethod("getApplication").apply {
                isAccessible = true
            }
            activityThread = currentActivityThread.invoke(null)
            application= getApplication.invoke(activityThread) as Context
        } catch (e: Throwable) {
            // 存在未适配的风险
        }
    }
    return application!!
}
复制代码

运行测试一下,context()的返回结果:

android.app.Application@c12661f
复制代码

2.3 小结

  • 优点:

依赖方不需要传递Context对象给库进行初始化,减少了代码耦合,有利于组件化

  • 缺点:

需要反射调用私有API,存在系统版本适配风险;使用反射有一定性能损耗


3. 使用 ContentProvider 获取 ApplicationContext

这一节介绍一种通过ContentProvider.java获得Application的方法。ContentProvider通常的用法是为当前进程 / 远程进程提供内容服务,它们会在应用启动的时候初始化,正因如此,我们可以利用ContentProvider来获得Context。

3.1 源码分析

ActivityThread.java

private void handleBindApplication(AppBindData data) {
    // ...
    Application app;
    app = data.info.makeApplication(data.restrictedBackupMode, null);
    mInitialApplication = app;
    // 初始化所有 ContentProvider
    installContentProviders(app, data.providers);
    // ...
}

private void installContentProviders(Context context, List<ProviderInfo> providers) {
    final ArrayList<ContentProviderHolder> results = new ArrayList<>();

    for (ProviderInfo cpi : providers) {
        // 依次初始化 ContentProvider
        ContentProviderHolder cph = installProvider(context, null, cpi,
                false /*noisy*/, true /*noReleaseNeeded*/, true /*stable*/);
        if (cph != null) {
            cph.noReleaseNeeded = true;
            results.add(cph);
        }
    }
    // ...
}
复制代码

可以看到,在ActivityThread中,创建Application对象之后会调用installContentProviders()安装属于当前进程(processName)的ContentProvider;而在ContentProvider声明的方法中,提供了getContext()获得ApplicationContext,所以,我们就可以ContentProvider启动的机制,从ContentProvider启动时拿到ApplicationContext

3.2 使用步骤

步骤一:实现 ContentProvider 子类

// ContextProvider.kt

internal class ContextProvider : ContentProvider(){

    override fun onCreate(): Boolean {
        init(context!!)
        return true
    }
    
    // 其他方法直接 return 
}
// Context.kt
private lateinit var application : Context

fun init(context : Context){
    application= context
}

fun context() : Context{
    return application
}
复制代码

步骤二:在 AndroidManifest 中配置

// AndroidManifest.xml

<application>
    <provider
        android:name=".Contextprovider"
        android:authorities="${applicationId}.contextprovider"
        android:exported="false" />
</application>
复制代码

步骤三:使用

Toast.makeText(context(),"",Toast.LENGTH_SHORT).show()
复制代码

3.3 小结

  • 优点:

依赖方不需要传递Context对象给库进行初始化,减少了代码耦合,有利于组件化

  • 缺点:

在App 启动时就初始化ContentProvider,不是懒初始化

  • 风险:

应保证初始化非常轻量,否则会降低App的启动速度


4. 案例

下面举出一些基于ContentProvider机制实现无侵入地获取Context的例子:

  • LeakCanary 2.4

AppWatcherInstaller.java

internal sealed class AppWatcherInstaller : ContentProvider() {

    internal class MainProcess : AppWatcherInstaller()

    internal class LeakCanaryProcess : AppWatcherInstaller()

    override fun onCreate(): Boolean {
        val application = context!!.applicationContext as Application
        AppWatcher.manualInstall(application)
        return true
    }

    // 其他方法直接 return 
}
复制代码
  • AutoSize 1.1.2

InitProvider.java

public class InitProvider extends ContentProvider {
    @Override
    public boolean onCreate() {
        AutoSizeConfig.getInstance()
                .setLog(true)
                .init((Application) getContext().getApplicationContext())
                .setUseDeviceSize(false);
        return true;
    }

    // 其他方法直接 return 
}
复制代码
  • Picasso 2.7

PicassoProvider.java

public final class PicassoProvider extends ContentProvider {

    @SuppressLint("StaticFieldLeak") static Context context;

    @Override public boolean onCreate() {
        context = getContext();
        return true;
    }

    // 其他方法直接 return 
}


链接:https://juejin.im/post/6887980244389593096
来源:掘金

相关推荐

为何越来越多的编程语言使用JSON(为什么编程)

JSON是JavascriptObjectNotation的缩写,意思是Javascript对象表示法,是一种易于人类阅读和对编程友好的文本数据传递方法,是JavaScript语言规范定义的一个子...

何时在数据库中使用 JSON(数据库用json格式存储)

在本文中,您将了解何时应考虑将JSON数据类型添加到表中以及何时应避免使用它们。每天?分享?最新?软件?开发?,Devops,敏捷?,测试?以及?项目?管理?最新?,最热门?的?文章?,每天?花?...

MySQL 从零开始:05 数据类型(mysql数据类型有哪些,并举例)

前面的讲解中已经接触到了表的创建,表的创建是对字段的声明,比如:上述语句声明了字段的名称、类型、所占空间、默认值和是否可以为空等信息。其中的int、varchar、char和decimal都...

JSON对象花样进阶(json格式对象)

一、引言在现代Web开发中,JSON(JavaScriptObjectNotation)已经成为数据交换的标准格式。无论是从前端向后端发送数据,还是从后端接收数据,JSON都是不可或缺的一部分。...

深入理解 JSON 和 Form-data(json和formdata提交区别)

在讨论现代网络开发与API设计的语境下,理解客户端和服务器间如何有效且可靠地交换数据变得尤为关键。这里,特别值得关注的是两种主流数据格式:...

JSON 语法(json 语法 priority)

JSON语法是JavaScript语法的子集。JSON语法规则JSON语法是JavaScript对象表示法语法的子集。数据在名称/值对中数据由逗号分隔花括号保存对象方括号保存数组JS...

JSON语法详解(json的语法规则)

JSON语法规则JSON语法是JavaScript对象表示法语法的子集。数据在名称/值对中数据由逗号分隔大括号保存对象中括号保存数组注意:json的key是字符串,且必须是双引号,不能是单引号...

MySQL JSON数据类型操作(mysql的json)

概述mysql自5.7.8版本开始,就支持了json结构的数据存储和查询,这表明了mysql也在不断的学习和增加nosql数据库的有点。但mysql毕竟是关系型数据库,在处理json这种非结构化的数据...

JSON的数据模式(json数据格式示例)

像XML模式一样,JSON数据格式也有Schema,这是一个基于JSON格式的规范。JSON模式也以JSON格式编写。它用于验证JSON数据。JSON模式示例以下代码显示了基本的JSON模式。{"...

前端学习——JSON格式详解(后端json格式)

JSON(JavaScriptObjectNotation)是一种轻量级的数据交换格式。易于人阅读和编写。同时也易于机器解析和生成。它基于JavaScriptProgrammingLa...

什么是 JSON:详解 JSON 及其优势(什么叫json)

现在程序员还有谁不知道JSON吗?无论对于前端还是后端,JSON都是一种常见的数据格式。那么JSON到底是什么呢?JSON的定义...

PostgreSQL JSON 类型:处理结构化数据

PostgreSQL提供JSON类型,以存储结构化数据。JSON是一种开放的数据格式,可用于存储各种类型的值。什么是JSON类型?JSON类型表示JSON(JavaScriptO...

JavaScript:JSON、三种包装类(javascript 包)

JOSN:我们希望可以将一个对象在不同的语言中进行传递,以达到通信的目的,最佳方式就是将一个对象转换为字符串的形式JSON(JavaScriptObjectNotation)-JS的对象表示法...

Python数据分析 只要1分钟 教你玩转JSON 全程干货

Json简介:Json,全名JavaScriptObjectNotation,JSON(JavaScriptObjectNotation(记号、标记))是一种轻量级的数据交换格式。它基于J...

比较一下JSON与XML两种数据格式?(json和xml哪个好)

JSON(JavaScriptObjectNotation)和XML(eXtensibleMarkupLanguage)是在日常开发中比较常用的两种数据格式,它们主要的作用就是用来进行数据的传...

取消回复欢迎 发表评论:

请填写验证码