1、 概念
Context中文含义为环境,上下文。从Android系统的角度来讲:Context就是用户与Andriod操作系统交互的一个过程,比如微信聊天,此时的“环境”是指聊天的界面以及相关数据的传输,Context在加载资源、启动Activity、获取系统服务、获取内部文件(夹)路径、创建View等操作时都需要参与。
2、 Context相关类的继承关系
由以上类的继承关系可以看出:Context是一个抽象类,Activity,Service,Application都继承自Context,从程序设计的角度来讲:Context是一个抽象类,Activity,Service,Application等都是该类的一个实现。
3、 相关类介绍:
Context类
定义路径:/frameworks/base/core/java/android/content/Context.java
说明:抽象类,定义了一组通用的API,提供了关于应用环境全局信息的接口, 它的执行被Android系统所提供。它允许获取以应用为特征的资源和类型,是一个统领一些资源(应用程序环境变量等)的上下文。
源代码(部分)如下:
/*
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.
*/
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);
…
}
ContextImpl类
定义路径 :/frameworks/base/core/java/android/app/ContextImpl.java
说明: Context类的实现类,实现了Context类的功能。该类的大部分方法都是直接调用。
源代码(部分)如下:
/**
*Common implementation of Context API, which provides the base context object for *Activity and other application components.
*/
Class ContextImpl extends Context{
//所有Application程序公用一个mPackageInfo对象
ActivityThread.PackageInfo mPackageInfo;
private Context mOuterContext;
@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引用,即ContextImpl对象的引用。
源代码(部分)如下:
/**
* Proxying implementation of Context that simply delegates all of its calls to
* another Context. Can be subclassed to modify behavior without changing
* the original Context.
*/
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类。
源代码(部分)如下:
/**
* A ContextWrapper that allows you to modify the theme from what is in the wrapped *context.
*/
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;
}
}
4、 Context对象的创建
由2、Context相关类的继承关系可知,Activity类 、Service类 、Application类本质上都是Context子类。因此应用程序需要创建Context实例有如下几种情况:
1、创建Application 对象时, 而且整个App只有一个Application对象
2、创建Service对象时
3、创建Activity对象时
因此应用程序App共有的Context数目公式为:
总Context实例个数 = Service个数 + Activity个数 + 1(Application对应的Context实例)
5、 Context对象创建时机
创建Application对象
每个应用程序在第一次启动时,都会首先创建Application对象。创建Application的时机在创建handleBindApplication()方法中,该函数位于 ActivityThread.java类中 ,如下:
//创建Application时同时创建的ContextImpl实例
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实例
}
...
}
创建Activity对象
通过startActivity()或startActivityForResult()请求启动一个新的Activity时,系统会回调handleLaunchActivity()方法,该方法内部会调用performLaunchActivity()方法,去创建一个Activity实例,然后回调Activity的onCreate()等方法,Activity的ContextImpl类的实例化是在 ActivityThread类的performLaunchActivity方法中 ,部分源码如下:
//创建一个Activity实例时同时创建ContextImpl实例
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实例
...
}
...
}
创建Service对象
通过startService或者bindService时,如果系统检测到需要新创建一个Service实例,就会回调handleCreateService()方法,完成相关数据操作。handleCreateService()函数位于 ActivityThread.java类,如下:
//创建一个Service实例时同时创建ContextImpl实例
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);
...
}
由以上可以看出,ComtextImpl类实现了Context类的所有功能,且Activity、Application、Service的创建都是在ActivityThread类中实现的。
另外,需要强调一点的是,通过对ContextIml的分析可知,其方法的大多数操作都是直接调用其属性mPackageInfo(该属性类型为PackageInfo)的相关方法而来。这说明ContextImp是一种轻量级类,而PackageInfo才是真正重量级的类。而一个App里的所有ContextImpl实例,都对应同一个packageInfo对象。
Android中Context可以有很多操作,其最主要的功能是加载和访问资源。在Android中常用两种Context,Application Context和Activity Context,通常我们在各种类和方法间传递的是Activity Context。