Context

作者: M_JCs | 来源:发表于2016-04-08 17:20 被阅读99次

    1、 概念

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

    2、 Context相关类的继承关系

    20151022212109519.png20151022212109519.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

        本文链接:https://www.haomeiwen.com/subject/sgpvlttx.html