彩票走勢圖網站是用什么程序做的搜索引擎優(yōu)化的目標
目錄
- 前言
- 一、Context簡介
- 二、Application Context
- 2.1 Application Context的創(chuàng)建過程
- 2.2 Application Context的獲取過程
- 三、Activity的Context創(chuàng)建過程
- 四、Service的Context創(chuàng)建過程
前言
Context也就是上下文對象,是Android較為常用的類,但是對于Context,大多都停留在會用的階段,本文會從源碼角度來分析Context,從而更加深入的理解它。
一、Context簡介
Context意為上下文或者場景,是一個應用程序環(huán)境信息的接口。
在開發(fā)中我們經常會使用Context,它的使用場景總的來說分為兩大類,它們分別是:
- 使用Context調用方法,比如:啟動Activity、訪問資源、調用系統(tǒng)級服務等。
- 調用方法時傳入Context,比如:彈出Toast、創(chuàng)建Dialog等。
Activity、Service和Application都是間接的繼承自Context的,因此,可以計算出一個應用程序進程中有多少個Context,這個數量等于Activity和Service的總個數加1,1指的是Application的數量。
Context是一個抽象類,它的內部定義了很多方法以及靜態(tài)常量,它的具體實現(xiàn)類為ContextImpl。和Context相關聯(lián)的類,除了ContextImpl還有ContextWrapper、ContextThemeWrapper和Activity等等,下面給出Context的關系圖。
從圖中我們可以看出,ContextImpl和ContextWrapper繼承自Context,ContextThemeWrapper、Service和Application繼承自ContextWrapper。ContextWrapper和ContextThemeWrapper都是Context的包裝類,它們都含有Context類型的mBase對象,mBase具體指向的是ContextImpl,這樣通過ContextWrapper和ContextThemeWrapper也可以使用Context的方法。ContextThemeWrapper中包含和主題相關的方法(比如: getTheme方法),因此,需要主題的Activity繼承ContextThemeWrapper,而不需要主題的Service則繼承ContextWrapper。
二、Application Context
2.1 Application Context的創(chuàng)建過程
我們通過調用getApplicationContext來獲取應用程序的全局的Application Context,那么Application Context是如何創(chuàng)建的呢?
當一個應用程序啟動完成后,應用程序就會有一個全局的Application Context。那么我們就從應用程序啟動過程開始著手。
ActivityThread作為應用程序進程的核心類,它會調用它的內部類ApplicationThread的scheduleLaunchActivity方法來啟動Activity,如下所示。
frameworks/base/core/java/android/app/ActivityThread.java
private class ApplicationThread extends ApplicationThreadNative {...@Overridepublic final void scheduleLaunchActivity(Intent intent, IBinder token, int ident,ActivityInfo info, Configuration curConfig, Configuration overrideConfig,CompatibilityInfo compatInfo, String referrer, IVoiceInteractor voiceInteractor,int procState, Bundle state, PersistableBundle persistentState,List<ResultInfo> pendingResults, List<ReferrerIntent> pendingNewIntents,boolean notResumed, boolean isForward, ProfilerInfo profilerInfo) {updateProcessState(procState, false);ActivityClientRecord r = new ActivityClientRecord();...sendMessage(H.LAUNCH_ACTIVITY, r);}... }
在ApplicationThread的scheduleLaunchActivity方法中向H類發(fā)送LAUNCH_ACTIVITY
類型的消息,目的是將啟動Activity的邏輯放在主線程中的消息隊列中,這樣啟動Activity的邏輯會在主線程中執(zhí)行。我們接著查看H類的handleMessage方法對LAUNCH_ACTIVITY類型的消息的處理。
frameworks/base/core/java/android/app/ActivityThread.java
private class H extends Handler {public static final int LAUNCH_ACTIVITY = 100;
...
public void handleMessage(Message msg) {if (DEBUG_MESSAGES) Slog.v(TAG, ">>> handling: " + codeToString(msg.what));switch (msg.what) {case LAUNCH_ACTIVITY: {Trace.traceBegin(Trace.TRACE_TAG_ACTIVITY_MANAGER, "activityStart");final ActivityClientRecord r = (ActivityClientRecord) msg.obj;r.packageInfo = getPackageInfoNoCheck(r.activityInfo.applicationInfo, r.compatInfo);//1handleLaunchActivity(r, null, "LAUNCH_ACTIVITY");//2Trace.traceEnd(Trace.TRACE_TAG_ACTIVITY_MANAGER);} break;...
}
H繼承自Handler ,是ActivityThread的內部類。在注釋1處通過getPackageInfoNoCheck方法獲得LoadedApk類型的對象,并將該對象賦值給ActivityClientRecord 的成員變量packageInfo,其中LoadedApk用來描述已加載的APK文件。在注釋2處調用handleLaunchActivity方法,如下所示。
frameworks/base/core/java/android/app/ActivityThread.java
private void handleLaunchActivity(ActivityClientRecord r, Intent customIntent, String reason) {...Activity a = performLaunchActivity(r, customIntent);...}
接著查看performLaunchActivity方法:
frameworks/base/core/java/android/app/ActivityThread.java
private Activity performLaunchActivity(ActivityClientRecord r, Intent customIntent) {...try {Application app = r.packageInfo.makeApplication(false, mInstrumentation);...} ...return activity;}
performLaunchActivity方法中有很多重要的邏輯,這里只保留了Application Context相關的邏輯,這里ActivityClientRecord 的成員變量packageInfo是LoadedApk類型的,接著來查看LoadedApk的makeApplication方法,如下所示。
frameworks/base/core/java/android/app/LoadedApk.java
public Application makeApplication(boolean forceDefaultAppClass,Instrumentation instrumentation) {if (mApplication != null) {//1return mApplication;}...try {...java.lang.ClassLoader cl = getClassLoader();...ContextImpl appContext = ContextImpl.createAppContext(mActivityThread, this);//2app = mActivityThread.mInstrumentation.newApplication(cl, appClass, appContext);//3appContext.setOuterContext(app);//4} catch (Exception e) {...}mActivityThread.mAllApplications.add(app);mApplication = app;//5...return app;
}
注釋1
處如果mApplication不為null則返回mApplication,這里假設是第一次啟動應用程序,因此mApplication為null。
在注釋2
處通過ContextImpl的createAppContext方法來創(chuàng)建ContextImpl。
注釋3
處的代碼用來創(chuàng)建Application,在Instrumentation的newApplication方法中傳入了ClassLoader類型的對象以及注釋2處創(chuàng)建的ContextImpl 。
在注釋4
處將Application賦值給ContextImpl的Context類型的成員變量mOuterContext。
注釋5
處將Application賦值給LoadedApk的成員變量mApplication,在Application Context的獲取過程中我們會再次用到mApplication。
來查看注釋3處的Application是如何創(chuàng)建的,Instrumentation的newApplication方法如下所示。
frameworks/base/core/java/android/app/Instrumentation.java
static public Application newApplication(Class<?> clazz, Context context)throws InstantiationException, IllegalAccessException, ClassNotFoundException {Application app = (Application)clazz.newInstance();//1app.attach(context);return app;
}
Instrumentation中有兩個newApplication重載方法,最終會調用上面這個重載方法。注釋1處通過反射來創(chuàng)建Application,并調用了Application的attach方法,并將ContextImpl傳進去:
frameworks/base/core/java/android/app/Application.java
/* package */ final void attach(Context context) {attachBaseContext(context);mLoadedApk = ContextImpl.getImpl(context).mPackageInfo;
}
attach方法中調用了attachBaseContext方法,它的實現(xiàn)在Application的父類ContextWrapper中,代碼如下所示。
frameworks/base/core/java/android/content/ContextWrapper.java
protected void attachBaseContext(Context base) {if (mBase != null) {throw new IllegalStateException("Base context already set");}mBase = base;}
從上文得知,這個base指的是ContextImpl,將ContextImpl賦值給ContextWrapper的Context類型的成員變量mBase。
2.2 Application Context的獲取過程
熟知了Application Context的創(chuàng)建過程,那么它的獲取過程會非常好理解。我們通過調用getApplicationContext方法來獲得Application Context,getApplicationContext方法的實現(xiàn)在ContextWrapper中,如下所示。
frameworks/base/core/java/android/content/ContextWrapper.java
@Overridepublic Context getApplicationContext() {return mBase.getApplicationContext();}
從上文得知,mBase指的是ContextImpl,我們來查看 ContextImpl的getApplicationContext方法:
frameworks/base/core/java/android/app/ContextImpl.java
Override
public Context getApplicationContext() {return (mPackageInfo != null) ?mPackageInfo.getApplication() : mMainThread.getApplication();
}
如果LoadedApk不為null,則調用LoadedApk的getApplication方法,否則調用AvtivityThread的getApplication方法。由于應用程序這時已經啟動,因此LoadedApk不會為null,則會調用LoadedApk的getApplication方法:
frameworks/base/core/java/android/app/LoadedApk.java
Application getApplication() {return mApplication;}
這里的mApplication我們應該很熟悉,它在上文LoadedApk的makeApplication方法的注釋5處被賦值。這樣我們通過getApplicationContext方法就獲取到了Application Context。
三、Activity的Context創(chuàng)建過程
當我們在Activity中調用startActivity方法時,其實調用的是Context的startActivity方法,如果想要在Activity中使用Context提供的方法,務必要先創(chuàng)建Context。Activity的Context會在Activity的啟動過程中被創(chuàng)建, ActivityThread是應用程序進程的核心類,它的內部類ApplicationThread會調用scheduleLaunchActivity方法來啟動Activity,scheduleLaunchActivity方法如下所示。
frameworks/base/core/java/android/app/ActivityThread.java
Override
public final void scheduleLaunchActivity(Intent intent, IBinder token, int ident,ActivityInfo info, Configuration curConfig, Configuration overrideConfig,CompatibilityInfo compatInfo, String referrer, IVoiceInteractor voiceInteractor,int procState, Bundle state, PersistableBundle persistentState,List<ResultInfo> pendingResults, List<ReferrerIntent> pendingNewIntents,boolean notResumed, boolean isForward, ProfilerInfo profilerInfo) {updateProcessState(procState, false);ActivityClientRecord r = new ActivityClientRecord();r.token = token;...sendMessage(H.LAUNCH_ACTIVITY, r);
}
scheduleLaunchActivity方法會將啟動Activity的參數封裝成ActivityClientRecord ,sendMessage方法向H類發(fā)送類型為LAUNCH_ACTIVITY
的消息,并將ActivityClientRecord 傳遞過去。sendMessage方法的目的是將啟動Activity的邏輯放在主線程中的消息隊列中,這樣啟動Activity的邏輯就會在主線程中執(zhí)行。
H類的handleMessage方法中會對LAUNCH_ACTIVITY類型的消息進行處理,其中調用了handleLaunchActivity方法,而handleLaunchActivity方法中又調用performLaunchActivity方法,來查看performLaunchActivity
方法。
frameworks/base/core/java/android/app/ActivityThread.java
private Activity performLaunchActivity(ActivityClientRecord r, Intent customIntent) {...Activity activity = null;try {java.lang.ClassLoader cl = r.packageInfo.getClassLoader();activity = mInstrumentation.newActivity(cl, component.getClassName(), r.intent);//1...}} catch (Exception e) {...}try {...if (activity != null) {Context appContext = createBaseContextForActivity(r, activity);//2.../***3*/activity.attach(appContext, this, getInstrumentation(), r.token,r.ident, app, r.intent, r.activityInfo, title, r.parent,r.embeddedID, r.lastNonConfigurationInstances, config,r.referrer, r.voiceInteractor, window); ...if (r.isPersistable()) {mInstrumentation.callActivityOnCreate(activity, r.state, r.persistentState);//4} else {mInstrumentation.callActivityOnCreate(activity, r.state);}...}return activity;}
performLaunchActivity方法中有很多重要的邏輯,這里只保留了Activity的Context相關的邏輯。在注釋1處用來創(chuàng)建Activity的實例。注釋2處通過createBaseContextForActivity方法用來創(chuàng)建Activity的ContextImpl,并將ContextImpl傳入注釋3處的activity的attach方法中。在注釋4處Instrumentation的callActivityOnCreate方法中會調用Activity的onCreate方法。
我們先來查看注釋2出的createBaseContextForActivity
方法:
frameworks/base/core/java/android/app/ActivityThread.java
private Context createBaseContextForActivity(ActivityClientRecord r, final Activity activity) {...ContextImpl appContext = ContextImpl.createActivityContext(this, r.packageInfo, r.token, displayId, r.overrideConfig);//1appContext.setOuterContext(activity);//2Context baseContext = appContext;...return baseContext;}
在注釋1處調用ContextImpl的createActivityContext方法來創(chuàng)建ContextImpl,注釋2處調用了ContextImpl的setOuterContext方法,將此前創(chuàng)建的Activity 實例賦值給ContextImpl的成員變量mOuterContext,這樣ContextImpl也可以訪問Activity的變量和方法。
我們再回到ActivityThread的performLaunchActivity方法,查看注釋3處的Activity的attach
方法,如下所示。
frameworks/base/core/java/android/app/Activity.java
final void attach(Context context, ActivityThread aThread,Instrumentation instr, IBinder token, int ident,Application application, Intent intent, ActivityInfo info,CharSequence title, Activity parent, String id,NonConfigurationInstances lastNonConfigurationInstances,Configuration config, String referrer, IVoiceInteractor voiceInteractor,Window window) {attachBaseContext(context);//1mFragments.attachHost(null /*parent*/);mWindow = new PhoneWindow(this, window);//2mWindow.setWindowControllerCallback(this);mWindow.setCallback(this);//3mWindow.setOnWindowDismissedCallback(this);...mWindow.setWindowManager((WindowManager)context.getSystemService(Context.WINDOW_SERVICE),mToken, mComponent.flattenToString(),(info.flags & ActivityInfo.FLAG_HARDWARE_ACCELERATED) != 0);//4if (mParent != null) {mWindow.setContainer(mParent.getWindow());}mWindowManager = mWindow.getWindowManager();//5mCurrentConfig = config;}
在注釋2處創(chuàng)建PhoneWindow,它代表應用程序窗口。PhoneWindow在運行中會間接觸發(fā)很多事件,比如點擊事件、菜單彈出、屏幕焦點變化等事件,這些事件需要轉發(fā)給與PhoneWindow關聯(lián)的Actvity,轉發(fā)操作通過Window.Callback接口實現(xiàn),Actvity實現(xiàn)了這個接口,在注釋3處將當前Activity通過Window的setCallback方法傳遞給PhoneWindow。
注釋4處給PhoneWindow設置WindowManager,并在注釋5處獲取WindowManager并賦值給Activity的成員變量mWindowManager ,這樣在Activity中就可以通過getWindowManager方法來獲取WindowManager。
在注釋1處調用了ContextThemeWrapper的attachBaseContext
方法,如下所示。
frameworks/base/core/java/android/view/ContextThemeWrapper.java
Override
protected void attachBaseContext(Context newBase) {super.attachBaseContext(newBase);
}
attachBaseContext方法接著調用ContextThemeWrapper的父類ContextWrapper的attachBaseContext
方法:
frameworks/base/core/java/android/content/ContextWrapper.java
protected void attachBaseContext(Context base) {if (mBase != null) {throw new IllegalStateException("Base context already set");}mBase = base;//1
}
注釋1處的base指的是一路傳遞過來的Activity的ContextImpl,將它賦值給ContextWrapper的成員變量mBase。這樣ContextWrapper的功能就可以交由ContextImpl處理,舉個例子:
frameworks/base/core/java/android/content/ContextWrapper.java
@Override
public Resources.Theme getTheme() {return mBase.getTheme();
}
當調用ContextWrapper的getTheme方法,其實就是調用的ContextImpl的getTheme方法。
Activity的Context創(chuàng)建過程就講到這里。 總結一下,在啟動Activity的過程中創(chuàng)建ContextImpl,并賦值給ContextWrapper的成員變量mBase中。Activity繼承自ContextWrapper的子類ContextThemeWrapper,這樣在Activity中就可以使用ContextImpl了。
四、Service的Context創(chuàng)建過程
Service的Context創(chuàng)建過程與Activity的Context創(chuàng)建過程類似,也是在Service的啟動過程中被創(chuàng)建 ActivityThread的內部類ApplicationThread會調用scheduleCreateService方法來啟動Service,如下所示。
frameworks/base/core/java/android/app/ActivityThread.java
public final void scheduleCreateService(IBinder token,ServiceInfo info, CompatibilityInfo compatInfo, int processState) {...sendMessage(H.CREATE_SERVICE, s);}
sendMessage方法向H類發(fā)送CREATE_SERVICE類型的消息,H類的handleMessage方法中會對CREATE_SERVICE類型的消息進行處理,其中調用了handleCreateService方法:
frameworks/base/core/java/android/app/ActivityThread.java
private void handleCreateService(CreateServiceData data) {...try {if (localLOGV) Slog.v(TAG, "Creating service " + data.info.name);ContextImpl context = ContextImpl.createAppContext(this, packageInfo);//1context.setOuterContext(service);Application app = packageInfo.makeApplication(false, mInstrumentation);service.attach(context, this, data.info.name, data.token, app,ActivityManagerNative.getDefault());//2service.onCreate();...} catch (Exception e) {... }}
在注釋1處創(chuàng)建了ContextImpl ,并將該ContextImpl傳入注釋2處service的attach方法中:
frameworks/base/core/java/android/app/Service.java
public final void attach(Context context,ActivityThread thread, String className, IBinder token,Application application, Object activityManager) {attachBaseContext(context);//1mThread = thread; // NOTE: unused - remove?mClassName = className;mToken = token;mApplication = application;mActivityManager = (IActivityManager)activityManager;mStartCompatibility = getApplicationInfo().targetSdkVersion< Build.VERSION_CODES.ECLAIR;}
注釋1處調用了ContextWrapper的attachBaseContext方法。
frameworks/base/core/java/android/content/ContextWrapper.java
protected void attachBaseContext(Context base) {if (mBase != null) {throw new IllegalStateException("Base context already set");}mBase = base;
}
attachBaseContext方法在前文已經講過,這里不再贅述。
Service的Context創(chuàng)建過程就講解到這里,它和Activity的Context創(chuàng)建過程類似。
參考鏈接:
深度詳解 Android 之 Context
Android Context完全解析,你所不知道的Context的各種細節(jié)