ApplicationContext、ActivityContext、ViewContext和ContextImpl(BaseContext)的区别

原创 2015年11月21日 12:31:53

ContextWrapper

  • ContextWrapper 是 ContextImpl(也就是BaseContext) 的包装,所有对 ContextWrapper 的操作都会用代理到 BaseContext 上。
  • Application和Activity和Service都继承自ContextWrapper
  • android.content.ContextWrapper#getApplicationContext 获得Application实例,和Application.this 是同一个对象。因此Activity.getApplicationContext == Application.getApplicationContext == Service.getApplicationContext == Application.this == Activity.getApplication
  • android.content.ContextWrapper#getBaseContext 都获得的是ContextImpl对象。

Application中BaseContext的创建过程:

  • android.app.LoadedApk#makeApplication,首先判断Application实例是否存在,如果存在则直接返回,不存在继续下面步骤。
  • 首先调用 android.app.ContextImpl#createAppContext 创建一个ContextImpl实例;
  • 调用 android.app.Instrumentation#newApplication 创建一个Application实例;
  • 调用 android.app.Application#attach->android.content.ContextWrapper#attachBaseContext 为Application的BaseContext赋值;
  • 最后调用 android.app.ContextImpl#setOuterContext 将Application实例赋值给ContextImpl的mOuterContext,这个mOuterContext的作用:
    • startActivity时,如果是ActivityContext的话,就不需要加 FLAG_ACTIVITY_NEW_TASK 标记之类的
    • 创建部分SystemService(如:android.content.Context#NOTIFICATION_SERVICE)时,也会用到mOuterContext。具体参考ContextImpl的static块。
  • 因为一个进程中只会有 Application的一个实例,因此Application的BaseContext也只会存在一个。

Activity的BaseContext的创建过程:

  • android.app.ActivityThread#performLaunchActivity
  • android.app.Instrumentation#newActivity 创建Activity实例,直接调用Activity的无参构造
  • android.app.ActivityThread#createBaseContextForActivity 为Activity创建BaseContext。
  • android.app.ContextImpl#createActivityContext 创建ContextImpl实例
  • 设置ContextImpl的OuterContext为Activity
  • android.app.Activity#attach->android.content.ContextWrapper#attachBaseContext 为Activity的BaseContext赋值;
  • 因此Activity的BaseContext每次都是新的。

Service的BaseContext的创建过程:

参考:android.app.ActivityThread#handleCreateService 和Activity类似,同样每次BaseContext都是新的。

ViewContext

参考:android.view.LayoutInflater#createViewFromTag
如果ViewParent使用了不同的Theme之类的,就会和ViewParent使用同一个Context,否则使用 LayoutInflater 实例化时的Context,不过一般ViewContext == Activity.this。ViewContext不可能是ContextImpl,因为在ContextImpl对LAYOUT_INFLATER_SERVICE的获得是通过getOuterContext,也就是获得的是Activity,Service或Application。

总结:

  • android.content.ContextWrapper#getBaseContext 都获得的是ContextImpl对象。
  • Activity.getApplicationContext == Application.getApplicationContext == Service.getApplicationContext == Application.this == Activity.getApplication
  • Application的BaseContext只会存在一个,而Activity和Service的BaseContext每次都是新的.
  • 通常情况下 ViewContext == Activity.this,特殊情况比如:使用LayoutInflater.from(ApplicationContext) inflate的View的ViewContext == Application.this。ViewContext不可能是ContextImpl。
版权声明:本文为博主原创文章,未经博主允许不得转载。

相关文章推荐

Android中ContextImpl源码分析(二)

1、背景 在前一篇文章中我们分析了Android系统中的Context类及其子类相关关系,以及各种Context子类的创建过程,大家有兴趣的可以去了解一下Android中Context源码分析(一)...

每周荐书:ES6、虚拟现实、物联网(评论送书)

每周荐书:ES6、虚拟现实、物联网(评论送书) 感谢大家对每周荐书栏目的支持,先公布下上期活动中奖名单 夕阳雨晴KK-NevenJava Web轻量级开发面试教程 monkey66da架构小白机器人P...

面试复习——Android工程师之Android面试大纲

文章会不定期更新,欢迎关注我的博客 Activity面试题 Fragment面试题 Service面试题 Broadcast Receiver面试题 WebView面试题 Binder面试题 Hand...

framework之 ContextImpl文件解析

ContextImpl Context的实现
  • ahjxly
  • ahjxly
  • 2014-03-08 17:39
  • 5855

Android中Context源码分析(一)

Context源码解析

Android Context完全解析,你所不知道的Context的各种细节

Context相信所有的Android开发人员基本上每天都在接触,因为它太常见了。但是这并不代表Context没有什么东西好讲的,实际上Context有太多小的细节并不被大家所关注,那么今天我们就来学...

Spring中BeanFactory和ApplicationContext的区别

spring中BeanFactory和ApplicationContext的区别。

Spring中ApplicationContext和beanfactory区别

BeanFacotry是spring中比较原始的Factory。如XMLBeanFactory就是一种典型的BeanFactory。原始的BeanFactory无法支持spring的许多插件,如AOP...

Spring的applicationContext.xml和dispatcher-servlet.xml的区别

大家知道, 在spring mvc中, 在applicationContext.xml 和 dispatch-servlet.xml中都可以进行spring 的配置, 那么他们有什么区别呢: 我们先...

Spring 学习手札(三)BeanFactory与ApplicationContext的区别

一、基础介绍: 1、在Spring中,组件无需自己负责与其他组件的关联。取而代之的是,容器负责把协作组件的引用给予各个组件。创建系统组件之间协作关系的这个动作是DI的关键,通常被称之为装配; 2、...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:深度学习:神经网络中的前向传播和反向传播算法推导
举报原因:
原因补充:

(最多只允许输入30个字)