Context应用上下文理解

文章目录

    • 一、Context相关类的继承关系
      • Context类
      • ContextIml.java类
      • ContextWrapper类
      • ContextThemeWrapper类
    • 二、 什么时候创建Context实例
      • 创建Context实例的时机
    • 小结

Context类 ,说它熟悉,是应为我们在开发中时刻的在与它打交道,例如:Service、BroadcastReceiver、Activity等都会利用到Context的相关方法。

下面开始分析和理解Context上下文的实现原理。

Context,中文直译为“上下文”,SDK中对其说明如下:

Interface to global information about an application environment. This is an abstract class whose implementation is provided by the Android system. It allows access to application-specific resources and classes, as well as up-calls for application-level operations such as launching activities, broadcasting and receiving intents, etc

从上可知一下三点,即:
1. 它描述的是一个应用程序环境的信息,即上下文。
2. 该类是一个抽象(abstract class)类,Android提供了该抽象类的具体实现类(后面我们会讲到是ContextIml类)。
3. 通过它我们可以获取应用程序的资源和类,也包括一些应用级别操作,例如:启动一个Activity,发送广播,接受Intent信息等。

于是,我们可以利用该Context对象去构建应用级别操作(application-level operations) 。

一、Context相关类的继承关系

Context类

路径: /frameworks/base/core/Java/android/content/Context.java
说明:抽象类,提供了一组通用的API。
源代码(部分)如下:

public abstract class Context {  ...  public abstract Object getSystemService(String name);  //获得系统级服务  public abstract void startActivity(Intent intent);     //通过一个Intent启动Activity  public abstract ComponentName startService(Intent service);  //启动Service  //根据文件名得到SharedPreferences对象  public abstract SharedPreferences getSharedPreferences(String name,int mode);  ...  
}  

ContextIml.java类

路径 :/frameworks/base/core/java/android/app/ContextImpl.java
说明:该Context类的实现类为ContextIml,该类实现了Context类的功能。请注意,该函数的大部分功能都是直接调用其属性mPackageInfo去完成,这点我们后面会讲到。
源代码(部分)如下:

/** * Common implementation of Context API, which provides the base * context object for Activity and other application components. */  
class ContextImpl extends Context{  //所有Application程序公用一个mPackageInfo对象  /*package*/ ActivityThread.PackageInfo mPackageInfo;  @Override  public Object getSystemService(String name){  ...  else if (ACTIVITY_SERVICE.equals(name)) {  return getActivityManager();  }   else if (INPUT_METHOD_SERVICE.equals(name)) {  return InputMethodManager.getInstance(this);  }  }   @Override  public void startActivity(Intent intent) {  ...  //开始启动一个Activity  mMainThread.getInstrumentation().execStartActivity(  getOuterContext(), mMainThread.getApplicationThread(), null, null, intent, -1);  }  
}  

ContextWrapper类

路径 :\frameworks\base\core\java\android\content\ContextWrapper.java
说明:正如其名称一样,该类只是对Context类的一种包装,该类的构造函数包含了一个真正的Context引用,即ContextIml对象。
源代码(部分)如下:

public class ContextWrapper extends Context {  Context mBase;  //该属性指向一个ContextIml实例,一般在创建Application、Service、Activity时赋值  //创建Application、Service、Activity,会调用该方法给mBase属性赋值  protected void attachBaseContext(Context base) {  if (mBase != null) {  throw new IllegalStateException("Base context already set");  }  mBase = base;  }  @Override  public void startActivity(Intent intent) {  mBase.startActivity(intent);  //调用mBase实例方法  }  
} 

ContextThemeWrapper类

路径:/frameworks/base/core/java/android/view/ContextThemeWrapper.java
说明:该类内部包含了主题(Theme)相关的接口,即android:theme属性指定的。只有Activity需要主题,Service不需要主题,所以Service直接继承于ContextWrapper类。
源代码(部分)如下:

public class ContextThemeWrapper extends ContextWrapper {  //该属性指向一个ContextIml实例,一般在创建Application、Service、Activity时赋值  private Context mBase;  //mBase赋值方式同样有一下两种  public ContextThemeWrapper(Context base, int themeres) {  super(base);  mBase = base;  mThemeResource = themeres;  }  @Override  protected void attachBaseContext(Context newBase) {  super.attachBaseContext(newBase);  mBase = newBase;  }  
} 
 Activity类 、Service类 、Application类本质上都是Context子类, 更多信息大家可以自行参考源代码进行理解。

二、 什么时候创建Context实例

熟悉了Context的继承关系后,我们接下来分析应用程序在什么情况需要创建Context对象的?应用程序创建Context实例的

情况有如下几种情况:

  1. 创建Application 对象时, 而且整个App共一个Application对象
  2. 创建Service对象时
  3. 创建Activity对象时
    因此应用程序App共有的Context数目公式为:
    总Context实例个数 = Service个数 + Activity个数 + 1(Application对应的Context实例)

创建Context实例的时机

1、创建Application对象的时机
每个应用程序在第一次启动时,都会首先创建Application对象。如果对应用程序启动一个Activity(startActivity)流程比较清楚的话,创建Application的时机在创建handleBindApplication()方法中,该函数位于 ActivityThread.java类中 ,如下:

//创建Application时同时创建的ContextIml实例  
private final void handleBindApplication(AppBindData data){  ...  ///创建Application对象  Application app = data.info.makeApplication(data.restrictedBackupMode, null);  ...  
}  
public Application makeApplication(boolean forceDefaultAppClass, Instrumentation instrumentation) {  ...  try {  java.lang.ClassLoader cl = getClassLoader();  ContextImpl appContext = new ContextImpl();    //创建一个ContextImpl对象实例  appContext.init(this, null, mActivityThread);  //初始化该ContextIml实例的相关属性  ///新建一个Application对象   app = mActivityThread.mInstrumentation.newApplication(  cl, appClass, appContext);  appContext.setOuterContext(app);  //将该Application实例传递给该ContextImpl实例           }   ...  
}

2、创建Activity对象的时机
通过startActivity()或startActivityForResult()请求启动一个Activity时,如果系统检测需要新建一个Activity对象时,就会回handleLaunchActivity()方法,该方法继而调用performLaunchActivity()方法,去创建一个Activity实例,并且回调onCreate(),onStart()方法等, 函数都位于ActivityThread.java类 ,如下:

//创建一个Activity实例时同时创建ContextIml实例  
private final void handleLaunchActivity(ActivityRecord r, Intent customIntent) {  ...  Activity a = performLaunchActivity(r, customIntent);  //启动一个Activity  
}  
private final Activity performLaunchActivity(ActivityRecord r, Intent customIntent) {  ...  Activity activity = null;  try {  //创建一个Activity对象实例  java.lang.ClassLoader cl = r.packageInfo.getClassLoader();  activity = mInstrumentation.newActivity(cl, component.getClassName(), r.intent);  }  if (activity != null) {  ContextImpl appContext = new ContextImpl();      //创建一个Activity实例  appContext.init(r.packageInfo, r.token, this);   //初始化该ContextIml实例的相关属性  appContext.setOuterContext(activity);            //将该Activity信息传递给该ContextImpl实例  ...  }  ...      
}  

3、创建Service对象的时机
通过startService或者bindService时,如果系统检测到需要新创建一个Service实例,就会回调handleCreateService()方法,完成相关数据操作.handleCreateService()函数位于 ActivityThread.java类,如下:

//创建一个Service实例时同时创建ContextIml实例  
private final void handleCreateService(CreateServiceData data){  ...  //创建一个Service实例  Service service = null;  try {  java.lang.ClassLoader cl = packageInfo.getClassLoader();  service = (Service) cl.loadClass(data.info.name).newInstance();  } catch (Exception e) {  }  ...  ContextImpl context = new ContextImpl(); //创建一个ContextImpl对象实例  context.init(packageInfo, null, this);   //初始化该ContextIml实例的相关属性  //获得我们之前创建的Application对象信息  Application app = packageInfo.makeApplication(false, mInstrumentation);  //将该Service信息传递给该ContextImpl实例  context.setOuterContext(service);  ...  
}  

小结

通过对ContextImp的分析可知,其方法的大多数操作都是直接调用其属性mPackageInfo(该属性类型为PackageInfo)的相关方法而来。这说明ContextImp是一种轻量级类,而PackageInfo才是真正重量级的类。而一个App里的所有ContextIml实例,都对应同一个packageInfo对象。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/95034.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

大数据-玩转数据-双流JOIN

一、双流JOIN 在Flink中, 支持两种方式的流的Join: Window Join和Interval Join 二、Window Join 窗口join会join具有相同的key并且处于同一个窗口中的两个流的元素. 注意: 1.所有的窗口join都是 inner join, 意味着a流中的元素如果在b流中没有对应的, 则a流中这个元素就不会…

棉花叶病害数据集

Bacterial Blight(细菌性枯萎病):细菌性枯萎病是由细菌引起的棉花疾病,主要受害部位是棉花的叶子和茎。这种病害可以导致叶片枯萎、变色和腐烂,对棉花产量产生不利影响。 Curl Virus(卷叶病毒)…

仿真调试说明——摘抄龙芯杯官方文件

1.仿真调试说明 你需要具备以下知识: 仿真工具的使用,比如Vivado的XsimVerilog的基本语法 通过本文的学习,你将获得:各类仿真错误排查的方法CPU逻辑出错的调试指导Verilog 运算符的优先级 1.1 调试指导思想概述 全局上的调试原…

多卡片效果悬停效果

效果展示 页面结构 从页面的结构上看&#xff0c;在默认状态下毛玻璃卡片是有层次感的效果叠加在一起&#xff0c;并且鼠标悬停在卡片区域后&#xff0c;卡片整齐排列。 CSS3 知识点 transform 属性的 rotate 值运用content 属性的 attr 值运用 实现页面整体布局 <div …

案例题--Web应用考点

案例题--Web应用考点 负载均衡技术微服务XML和JSON无状态和有状态真题 在选择题中没有考察过web的相关知识&#xff0c;主要就是在案例分析题中考察 负载均衡技术 应用层负载均衡技术 传输层负载均衡技术 就近的找到距离最近的服务器&#xff0c;并进行分发 使用户就近获取…

S32K144 GPIO编程

前面的文章介绍了如何在MDK-Keil下面进行S32K144的开发&#xff0c;下面就使用该工程模板进行GPIO LED的编程试验。 1. 开发环境 S32K144EVB-Q100开发板MDK-Keil Jlink 2. 硬件连接 S32K144EVB-Q100开发板关于LED的原理图如下&#xff1a; 也就是具体连接关系如下&#xf…

【C++】vector相关OJ

文章目录 1. 只出现一次的数字2. 杨辉三角3. 电话号码字母组合 ヾ(๑╹◡╹)&#xff89;" 人总要为过去的懒惰而付出代价ヾ(๑╹◡╹)&#xff89;" 1. 只出现一次的数字 力扣链接 代码展示&#xff1a; class Solution { public:int singleNumber(vector<i…

commons-io工具类常用方法

commons-io是Apache Commons项目的一个模块&#xff0c;提供了一系列处理I/O&#xff08;输入/输出&#xff09;操作的工具类和方法。它旨在简化Java I/O编程&#xff0c;并提供更多的功能和便利性。 读取文件内容为字符串 String path"C:\\Users\\zhang\\Desktop\\myyii\…

6-1 选择排序

#include <stdio.h>#define N 1000 int arr[N];/* 对长度为n的数组arr执行选择排序 */ void selectionSort(int arr[], int n);/* 打印长度为n的数组arr */ void printArray(int arr[], int n);void swap(int *xp, int *yp) {int temp *xp;*xp *yp;*yp temp; }int mai…

《C和指针》笔记32:多维数组初始化

文章目录 使用括号进行初始化初始化省略维度 使用括号进行初始化 我们可以给数组赋值一个长长的列表&#xff1a; int matrix[2][3] { 100, 101, 102, 110, 111, 112 };它等价于 matrix[0][0]100; matrix[0][1]101; matrix[0][2]102; matrix[1][0]110; matrix[1][1]111; ma…

数据库编程

数据库编程 文章目录 第一节 存储过程一、存储过程的基本概念二、创建存储过程三、存储过程体1、局部变量2、SET语句3、SELECT...INTO语句4、流程控制语句&#xff08;1&#xff09;IF语句&#xff08;2&#xff09;CASE语句&#xff08;3&#xff09;循环语句 5、游标&#xf…

uniapp iOS离线打包——如何创建App并提交版本审核?

uniapp 如何创建App&#xff0c;并提交版本审核&#xff1f; 文章目录 uniapp 如何创建App&#xff0c;并提交版本审核&#xff1f;登录 appstoreconnect创建AppiOS 预览和截屏应用功能描述技术支持App 审核信息 App 信息内容版权年龄分级 价格与销售范围App 隐私提交审核 登录…

华为云云耀云服务器L实例评测|安装搭建学生成绩管理系统

1.前言概述 华为云耀云服务器L实例是新一代开箱即用、面向中小企业和开发者打造的全新轻量应用云服务器。多种产品规格&#xff0c;满足您对成本、性能及技术创新的诉求。云耀云服务器L实例提供丰富严选的应用镜像&#xff0c;实现应用一键部署&#xff0c;助力客户便捷高效的在…

mybatis-plus控制台打印sql(mybatis-Log)

配置了mybatis-plus.configuration.log-implorg.apache.ibatis.logging.stdout.StdOutImpl&#xff1b;但是mybatis执行的sql没有输出 需要检查点&#xff1a; 1、日志级别设置&#xff1a;请确保你的日志级别配置正确。如果日志级别设置得太低&#xff0c;可能导致SQL语句不…

计算机竞赛 题目:基于LSTM的预测算法 - 股票预测 天气预测 房价预测

文章目录 0 简介1 基于 Keras 用 LSTM 网络做时间序列预测2 长短记忆网络3 LSTM 网络结构和原理3.1 LSTM核心思想3.2 遗忘门3.3 输入门3.4 输出门 4 基于LSTM的天气预测4.1 数据集4.2 预测示例 5 基于LSTM的股票价格预测5.1 数据集5.2 实现代码 6 lstm 预测航空旅客数目数据集预…

【kubernetes】CRI OCI

1 OCI OCI(Open Container Initiative)&#xff1a;由Linux基金会主导&#xff0c;主要包含容器镜像规范和容器运行时规范&#xff1a; Image Specification(image-spec)Runtime Specification(runtime-spec)runC image-spec定义了镜像的格式&#xff0c;镜像的格式有以下几…

解决Qt msvc编译器 中文显示乱码问题

第一步&#xff1a;代码文件选择用utf8编码带bom。第二步&#xff1a;在有中文汉字的代码文件顶部加一行&#xff08;一般是cpp文件&#xff09; #pragma execution_character_set(“utf-8”) 可以考虑放在head.h中&#xff0c;然后需要的地方就引入head头文件就行&#xff0c;…

【多级缓存】

文章目录 1. JVM进程缓存2. Lua语法3. 实现多级缓存3.1 反向代理流程3.2 OpenResty快速入门 4. 查询Tomcat4.1 发送http请求的API4.2 封装http工具4.3 基于ID负载均衡4.4 流程小结 5. Redis缓存查询5.1 实现Redis查询 6. Nginx本地缓存6.1 本地缓存API6.2 实现本地缓存查询 7. …

算法之数论知识点总结

文章目录 前言【校招笔试】数论知识点整数分块 前言 最近在准备秋招的时候发现&#xff0c;一些公司第一题会出数论相关的题目&#xff0c;这对于没打过acm类竞赛的人来说&#xff0c;基本上是不可能AC的&#xff0c;所以本篇就尽量以最短的篇幅概括遇到的各种问题的结论&…

Django的模版使用(Django-03)

一 模版的使用 模板引擎是一种可以让开发者把服务端数据填充到html网页中完成渲染效果的技术。它实现了 把前端代码和服务端代码分离 的作用&#xff0c;让项目中的业务逻辑代码和数据表现代码分离&#xff0c;让前端开发者和服务端开发者可以更好的完成协同开发。 静态网页&…