平時我們加載xml文件都是直接在onCreate方法中調用setContentView棍丐,在加載Activity的onCreate方法時布局就被加載出來了差导,但是深入一點看郭怪,發(fā)現(xiàn)內(nèi)容還是很多的,看了很多大神相關的博客粘勒,也寫了個總結鹤树,內(nèi)容可能不全击费,不足之處俏让,還請多指教。
1.先從setContentView開始說起
Activity中有3個setContentView()
方法鳍徽,可以看到资锰, 這三個方法都是先getWindow
,獲得一個Window對象,然后調用它的setContentView
阶祭,那么這個Window又是什么呢绷杜?
public void setContentView(int layoutResID) {
getWindow().setContentView(layoutResID);
initWindowDecorActionBar();
}
///
public void setContentView(View view) {
getWindow().setContentView(view);
initWindowDecorActionBar();
}
////
public void setContentView(View view, ViewGroup.LayoutParams params) {
getWindow().setContentView(view, params);
initWindowDecorActionBar();
}
1.1 創(chuàng)建Window對象
getWindow返回的是mWindow,而這個mWindow是由PolicyManager創(chuàng)建的胖翰,PolicyManager提供了靜態(tài)類方法(這里用到了工廠模式接剩,PolicyManager提供工廠方法),創(chuàng)建了一個PhoneWindow 對象萨咳。
//創(chuàng)建一個Window對象
mWindow = PolicyManager.makeNewWindow(this);
最終創(chuàng)建Window對象的方法
//創(chuàng)建具體對象的接口
public PhoneWindow makeNewWindow(Context context) {
return new PhoneWindow(context);
}
Window:是一個抽象類懊缺,提供繪制窗口的通用API。
PhoneWindow :是Window的唯一的實現(xiàn)類培他,每個Activity都會有一個PhoneWindow鹃两,它是Activity和整個View交互的接口。該類內(nèi)部包含了一個DecorView對象舀凛,該DectorView對象是所有應用窗口(Activity界面)的根View俊扳。
DectorView:是PhoneWindow的內(nèi)部類,繼承自FrameLayout猛遍。
1.2 調用PhoneWindow對象的setContentView方法
一層層下來馋记,發(fā)現(xiàn)Activity的setContentView()
實際上是執(zhí)行的是PhoneWindow的方法,現(xiàn)在來看一下PhoneWindow的setContentView()
懊烤。
@Override
public void setContentView(int layoutResID) {
// Note: FEATURE_CONTENT_TRANSITIONS may be set in the process of installing the window
// decor, when theme attributes and the like are crystalized. Do not check the feature
// before this happens.
if (mContentParent == null) {
installDecor();
} else if (!hasFeature(FEATURE_CONTENT_TRANSITIONS)) {
mContentParent.removeAllViews(); }
if (hasFeature(FEATURE_CONTENT_TRANSITIONS)) {
final Scene newScene = Scene.getSceneForLayout(mContentParent, layoutResID, getContext());
transitionTo(newScene);
} else {
mLayoutInflater.inflate(layoutResID, mContentParent);
}
final Callback cb = getCallback();
if (cb != null && !isDestroyed()) {
cb.onContentChanged();
}
}
以上代碼分析:
1 . 判斷mContentParent
是否為空梯醒。從名字可以看出,這是父容器腌紧,它是一個ViewGroup類型的對象茸习,是真正的content的Parent,如果是第一次調用壁肋,會調用installDecor()
号胚,在這個方法中會先判斷DectorView是否為空籽慢,為空就先創(chuàng)建DectorView對象mDecor,然后調用generateLayout(mDecor)
得到mContentParent
猫胁;如果不是第一次調用箱亿,會先清除mContentParent
中的子view。
2 . mLayoutInflater.inflate(layoutResID, mContentParent);
將傳入的資源文件轉換成View樹杜漠,再添加到mContentParent中(mLayoutInflater 在PhoneWindow的構造函數(shù)中通過mLayoutInflater = LayoutInflater.from(context)
得到)极景。
再來看一下其他的兩個setContentView()
:
@Override
public void setContentView(View view) {
setContentView(view, new ViewGroup.LayoutParams(MATCH_PARENT, MATCH_PARENT));
}
@Override
public void setContentView(View view, ViewGroup.LayoutParams params) {
// Note: FEATURE_CONTENT_TRANSITIONS may be set in the process of installing the window
// decor, when theme attributes and the like are crystalized. Do not check the feature
// before this happens.
if (mContentParent == null) {
installDecor();
} else if (!hasFeature(FEATURE_CONTENT_TRANSITIONS)) {
mContentParent.removeAllViews();
}
if (hasFeature(FEATURE_CONTENT_TRANSITIONS)) {
view.setLayoutParams(params);
final Scene newScene = new Scene(mContentParent, view);
transitionTo(newScene);
} else {
mContentParent.addView(view, params);
}
final Callback cb = getCallback();
if (cb != null && !isDestroyed()) {
cb.onContentChanged();
}
}
可以看到,其實做的事情都差不多驾茴,只不過多設置了LayoutParams
,因為傳入的View氢卡,就不需要像第一個那樣還從xml文件中解析出來锈至,可以直接將view添加到mContentParent
中。
小結:比較3個setContentView
從上面的代碼中译秦,我們可以看到峡捡,第一個setContentView
是通過反射解析傳入的布局文件,然后添加到mContentParent
筑悴,而后兩個setContentView
是直接將傳入的View添加到mContentParent
们拙,需要注意的是,每次反射拿到的View都是重新創(chuàng)建的阁吝,就算兩次setContentView加載的是同一個布局文件砚婆,控件的實例也是不一樣的,如傳入的是View/ViewGroup就能保證傳入的是同一組控件突勇。
1.3 installDecor()實例化DectorView對象
現(xiàn)在來看一下剛剛提到的installDecor()
装盯,初始化mDecor ,創(chuàng)建mContentParent
甲馋,根據(jù)窗口的風格修飾埂奈,選擇對應的修飾布局文件,這里內(nèi)容太多定躏,省略了账磺。
private void installDecor() {
if (mDecor == null) {
mDecor = generateDecor();
mDecor.setDescendantFocusability(ViewGroup.FOCUS_AFTER_DESCENDANTS);
mDecor.setIsRootNamespace(true);
if (!mInvalidatePanelMenuPosted && mInvalidatePanelMenuFeatures != 0) {
mDecor.postOnAnimation(mInvalidatePanelMenuRunnable);
}
}
if (mContentParent == null) {
//根據(jù)窗口的風格修飾,選擇對應的修飾布局文件
mContentParent = generateLayout(mDecor);
......
}
}
1.4 generateLayout()創(chuàng)建mContentParent
接下來看generateLayout()
創(chuàng)建mContentParent 的過程痊远。
protected ViewGroup generateLayout(DecorView decor) {
// Apply data from current theme.
TypedArray a = getWindowStyle();
//根據(jù)當前的主題設置窗口屬性
......
// Inflate the window decor.
int layoutResource;
int features = getLocalFeatures();
//根據(jù)當前的窗口屬性選擇相對應的布局
WindowManager.LayoutParams params = getAttributes();
......
//將相應的布局文件轉成view添加到窗口視圖對象decor中
View in = mLayoutInflater.inflate(layoutResource, null);
decor.addView(in, new ViewGroup.LayoutParams(MATCH_PARENT, MATCH_PARENT));
mContentRoot = (ViewGroup) in;
ViewGroup contentParent = (ViewGroup)findViewById(ID_Android_CONTENT);
if (contentParent == null) {
throw new RuntimeException("Window couldn't find content container view");
}
......
return contentParent;
}
這段代碼所做的事情:
- 根據(jù)當前的主題設置窗口屬性垮抗;
- 根據(jù)當前的窗口屬性選擇相對應的布局;
- 將相應的布局文件轉成view添加到根視圖對象decor/mDecor中
View in = mLayoutInflater.inflate(layoutResource, null); decor.addView(in, new ViewGroup.LayoutParams(MATCH_PARENT, MATCH_PARENT));
拗引,需要注意的是借宵,這里的layoutResource并不是我們傳入的資源文件,而是系統(tǒng)定義的矾削。 - 從根布局中找到id為
ID_Android_CONTENT
的ViewGroup賦值給contentParent
壤玫,也就是上文的mContentParent
豁护。
總結:Activity,PhoneWindow欲间,DectorView楚里,mContentParent之間的關系
通過上面的分析,我們來看一下彼此之間的關系猎贴,有助于理解班缎。
DecorView繼承于FrameLayout,然后它有一個子view即LinearLayout她渴,方向為豎直方向达址,其內(nèi)有兩個FrameLayout,上面的FrameLayout即為TitleBar之類的趁耗,下面的FrameLayout即為我們的ContentView沉唠,所謂的setContentView就是往這個FrameLayout里面添加我們的布局View的!
2.PhoneWindow的setContentView最后的回調
上面分析了加載視圖到父容器mContentParent
中苛败,現(xiàn)在我們看一下setContentView()
中的最后一步满葛。
@Override
public void setContentView(int layoutResID) {
......
final Callback cb = getCallback();
if (cb != null && !isDestroyed()) {
cb.onContentChanged();
}
}
至此,已經(jīng)分析完了將傳入的布局文件添加到View的整個過程罢屈。接下來可以看到嘀韧,它先創(chuàng)建了一個CallBack回調接口,在加載完以上的View到根布局之后缠捌,就會調用這個回調接口锄贷,順便說一下,cb.onContentChanged()
方法在Activity中是一個空方法鄙币,我們可以在自定義的Activity中覆寫這個方法肃叶。
現(xiàn)在看getCallback()
是由Window提供的,PhoneWindow并沒有實現(xiàn)十嘿,繼續(xù)往下看因惭,發(fā)現(xiàn)Window中有一個public void setCallback(Callback callback)
方法,接收到外部傳入的callback绩衷,賦值給內(nèi)部的mCallback 蹦魔。那么這個外部方法在哪里調用呢?這個就要說一下Activity的啟動了咳燕。
3.Activity的啟動
在Activity加載時會先創(chuàng)建一個activity實例勿决,然后調用activity的attach方法完成activity的初始化過程,我們來看一下attach()
方法招盲。
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, IVoiceInteractor voiceInteractor) {
attachBaseContext(context);
mFragments.attachActivity(this, mContainer, null);
//1.創(chuàng)建窗口對象低缩,是一個PhoneWindow實例
mWindow = PolicyManager.makeNewWindow(this);
//2.設置回調
mWindow.setCallback(this);
mWindow.setOnWindowDismissedCallback(this);
mWindow.getLayoutInflater().setPrivateFactory(this);
......
mToken = token;
//3.將創(chuàng)建的WindowManager注入窗口對象以便管理窗口的視圖對象
mWindow.setWindowManager(
(WindowManager)context.getSystemService(Context.WINDOW_SERVICE),
mToken, mComponent.flattenToString(),
(info.flags & ActivityInfo.FLAG_HARDWARE_ACCELERATED) != 0);
if (mParent != null) {
mWindow.setContainer(mParent.getWindow());
}
//4. 獲取窗口管理器
mWindowManager = mWindow.getWindowManager();
mCurrentConfig = config;
}
在這個方法中,完成了上文提到的Window對象的創(chuàng)建。
為Window對象設置回調咆繁,也就是上面
setContentView()
中最后獲取到cb讳推。這里設置的回調就是Activity自己,再看Activity玩般,它實現(xiàn)了Window.Callback, KeyEvent.Callback
兩個回調接口银觅。其中KeyEvent.Callback
接口中聲明了處理手勢事件的方法(onKeyDown按下,onKeyUp抬起坏为,onKeyLongPress長按...)究驴,而Window.Callback
聲明了一些事件分發(fā)的函數(shù),關于View的事件分發(fā)匀伏,可以看這個 View的事件分發(fā)機制 ,從這里可以看出activity本身不具備處理用戶的事件的能力洒忧。為mWindow設置
WindowManager
,WindowManager
主要用來管理窗口的一些狀態(tài)帘撰、屬性跑慕、view增加、刪除摧找、更新、窗口順序牢硅、消息收集和處理等.獲取Window的WindowManager的實現(xiàn)類WindowManagerImpl保存在activity的mWindowManager中蹬耘。
3.1 關于mWindow.setWindowManager()方法
從setWindowManager方法中可以發(fā)現(xiàn),這里返回的是WindowManagerImpl
對象减余,WindowManager
是一個接口综苔,而WindowManagerImpl
是它的實現(xiàn)類,這里調用createLocalWindowManager()
位岔。
public void setWindowManager(WindowManager wm, IBinder appToken, String appName,
boolean hardwareAccelerated) {
mAppToken = appToken;
mAppName = appName;
mHardwareAccelerated = hardwareAccelerated
|| SystemProperties.getBoolean(PROPERTY_HARDWARE_UI, false);
if (wm == null) {
wm = (WindowManager)mContext.getSystemService(Context.WINDOW_SERVICE);
}
mWindowManager = ((WindowManagerImpl)wm).createLocalWindowManager(this);
}
WindowManagerImpl中的createLocalWindowManager()
方法
public WindowManagerImpl createLocalWindowManager(Window parentWindow) {
return new WindowManagerImpl(mDisplay, parentWindow);
}
3.2 將生成的窗口視圖對象是添加到手機屏幕
我們知道如筛,Activity的視圖是在Activity的生命周期onResume()
方法執(zhí)行之后才會顯示的,這是因為在ActivityThread的handleResumeActivity
方法中調用了Activity的onResume()
方法抒抬,關于Activity的啟動這一部分內(nèi)容我還沒有看過杨刨,“老羅的Android之旅”中有這一部分內(nèi)容的詳細分析,有興趣的可以看一下擦剑。在這個函數(shù)中妖胀,會調用activity的makeVisible方法經(jīng)WindowManagerImpl將DecorView展示出來。這里的getWindowManager()
就是之前設置的WindowManager惠勒。最后就Activity就可以請求WindowManagerService將視圖繪制到屏幕上了赚抡。
void makeVisible() {
if (!mWindowAdded) {
ViewManager wm = getWindowManager();
wm.addView(mDecor, getWindow().getAttributes());
mWindowAdded = true;
}
mDecor.setVisibility(View.VISIBLE);
}
總結
總結一下整個View的加載過程:
首先在Activity啟動時,在
attach
方法中先創(chuàng)建Activity的窗口對象纠屋,它是PhoneWindow
類型涂臣,每個Activity都有一個窗口對象,然后為這個窗口設置各種事件的回調售担,還要注冊其對應的窗口管理器赁遗,用來管理窗口的一些狀態(tài)署辉,屬性,view的更新等吼和。當調用Activity的
onCreat
方法時涨薪,會調用設置布局文件,Activity的setContentView
其實調用的是Activity的窗口對象PhoneWindow的setContentView炫乓,PhoneWindow有一個內(nèi)部類DectorView(FrameLayout的子類)
刚夺,它是整個窗口下的根View,內(nèi)部包含兩個FrameLayout末捣,一個根據(jù)主題樣式來進行TitleBar之類設置侠姑,一個就是用來裝我們傳入的布局文件中的view,這個就是mContetntParent
箩做。第一次調用PhoneWindow
的setcontentView
方法會先創(chuàng)建DectorView
莽红,進行一些初始化的設置,然后解析系統(tǒng)的資源文件到DectorView
中邦邦,接著會找到id為ID_Android_CONTENT
的FrameLayout
安吁,將其賦值給mContetntParent
,用來放我們傳入的資源文件解析出來的view.這些初始化的設置完成之后燃辖,就是處理我們調用
setContentView
時傳入的布局文件了鬼店,如果傳入的資源文件id,會調用反射機制解析xml文件黔龟,再把解析出來的各個view加到mContetntParent
妇智,如果傳入的是View
,那么直接加到mContetntParent
就可以氏身,最后就是系統(tǒng)在調用onResume
之后巍棱,經(jīng)之前設置的WindowManager
將整個DecorView
展示出來。