国产亚洲精品福利在线无卡一,国产精久久一区二区三区,亚洲精品无码国模,精品久久久久久无码专区不卡

當前位置: 首頁 > news >正文

彩票走勢圖網站是用什么程序做的搜索引擎優(yōu)化的目標

彩票走勢圖網站是用什么程序做的,搜索引擎優(yōu)化的目標,文件管理,廣告公司屬于什么行業(yè)目錄 前言一、Context簡介二、Application Context2.1 Application Context的創(chuàng)建過程2.2 Application Context的獲取過程 三、Activity的Context創(chuàng)建過程四、Service的Context創(chuàng)建過程 前言 Context也就是上下文對象,是Android較為常用的類,但是對于Co…

目錄

  • 前言
  • 一、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é)

http://m.aloenet.com.cn/news/28146.html

相關文章:

  • 海外 網站 推廣百度一下百度知道
  • 網站是否被百度收錄網址信息查詢
  • 怎嗎做網站掙錢揭陽seo快速排名
  • 如何用騰訊云做網站seo百度關鍵詞優(yōu)化
  • 聊城做網站好的公司淘寶網店代運營正規(guī)公司
  • 做網站美工廣州seo推廣公司
  • 山西傳染病最新消息今天唐山seo排名外包
  • 如何建設網站方便后期維護東莞網絡推廣公司
  • 做培訓的網站建設網絡推廣工作內容怎么寫
  • 游戲軟件開發(fā)屬于什么專業(yè)seo內部優(yōu)化具體做什么
  • 宣城市網站集約化建設茶葉網絡推廣方案
  • 長沙做企業(yè)網站的公司中國網站排名
  • 網站制作難點故事式的軟文廣告例子
  • 青島網站建設青島新思維百度用戶服務中心人工電話
  • 浙江省建設通網站迅雷磁力
  • 網站建設找星火龍關鍵詞排名代做
  • PC端網站開發(fā)以及設計費用qq群推廣網站
  • 網站建設公司排行榜搜索引擎優(yōu)化內容包括哪些方面
  • 貞豐縣住房和城鄉(xiāng)建設局網站鄭州關鍵詞排名顧問
  • 成都設計公司官網東莞seo網絡營銷
  • 泉州建設工程質量網站千鋒教育學費一覽表
  • 創(chuàng)建了網站百度推廣開戶費用
  • 小白學做網站買什么書優(yōu)化網站排名工具
  • wordpress查看網站內容站長工具服務器查詢
  • wordpress 操作數據庫大連seo顧問
  • 唐山企業(yè)網站建設濟南百度
  • 云主機 做網站鏈接交換公司
  • 上海網站搜索排名婚戀網站排名前三
  • 做外貿網站好的公司志鴻優(yōu)化設計答案網
  • 網站產品詳情用哪個軟件做的sem競價培訓班