Context

1、 概念

Context中文含义为环境,上下文。从Android系统的角度来讲:Context就是用户与Andriod操作系统交互的一个过程,比如微信聊天,此时的“环境”是指聊天的界面以及相关数据的传输,Context在加载资源、启动Activity、获取系统服务、获取内部文件(夹)路径、创建View等操作时都需要参与。

2、 Context相关类的继承关系

20151022212109519.png

由以上类的继承关系可以看出: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。

你可能感兴趣的:(Context)