View的繪制源碼分析

平時我們加載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;
}

這段代碼所做的事情:

  1. 根據(jù)當前的主題設置窗口屬性垮抗;
  2. 根據(jù)當前的窗口屬性選擇相對應的布局;
  3. 將相應的布局文件轉成view添加到根視圖對象decor/mDecor中
    View in = mLayoutInflater.inflate(layoutResource, null); decor.addView(in, new ViewGroup.LayoutParams(MATCH_PARENT, MATCH_PARENT));拗引,需要注意的是借宵,這里的layoutResource并不是我們傳入的資源文件,而是系統(tǒng)定義的矾削。
  4. 從根布局中找到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的!

DectorView及其下層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;
}
  1. 在這個方法中,完成了上文提到的Window對象的創(chuàng)建。

  2. 為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本身不具備處理用戶的事件的能力洒忧。

  3. 為mWindow設置WindowManagerWindowManager主要用來管理窗口的一些狀態(tài)帘撰、屬性跑慕、view增加、刪除摧找、更新、窗口順序牢硅、消息收集和處理等.

  4. 獲取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的加載過程:

  1. 首先在Activity啟動時,在attach方法中先創(chuàng)建Activity的窗口對象纠屋,它是PhoneWindow類型涂臣,每個Activity都有一個窗口對象,然后為這個窗口設置各種事件的回調售担,還要注冊其對應的窗口管理器赁遗,用來管理窗口的一些狀態(tài)署辉,屬性,view的更新等吼和。

  2. 當調用Activity的onCreat方法時涨薪,會調用設置布局文件,Activity的setContentView其實調用的是Activity的窗口對象PhoneWindow的setContentView炫乓,PhoneWindow有一個內(nèi)部類DectorView(FrameLayout的子類)刚夺,它是整個窗口下的根View,內(nèi)部包含兩個FrameLayout末捣,一個根據(jù)主題樣式來進行TitleBar之類設置侠姑,一個就是用來裝我們傳入的布局文件中的view,這個就是mContetntParent箩做。第一次調用PhoneWindowsetcontentView方法會先創(chuàng)建DectorView莽红,進行一些初始化的設置,然后解析系統(tǒng)的資源文件到DectorView中邦邦,接著會找到id為ID_Android_CONTENTFrameLayout安吁,將其賦值給mContetntParent,用來放我們傳入的資源文件解析出來的view.

  3. 這些初始化的設置完成之后燃辖,就是處理我們調用setContentView時傳入的布局文件了鬼店,如果傳入的資源文件id,會調用反射機制解析xml文件黔龟,再把解析出來的各個view加到mContetntParent妇智,如果傳入的是View,那么直接加到mContetntParent就可以氏身,最后就是系統(tǒng)在調用onResume之后巍棱,經(jīng)之前設置的WindowManager將整個DecorView展示出來。

最后編輯于
?著作權歸作者所有,轉載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末蛋欣,一起剝皮案震驚了整個濱河市航徙,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌豁状,老刑警劉巖捉偏,帶你破解...
    沈念sama閱讀 206,723評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異泻红,居然都是意外死亡夭禽,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,485評論 2 382
  • 文/潘曉璐 我一進店門谊路,熙熙樓的掌柜王于貴愁眉苦臉地迎上來讹躯,“玉大人,你說我怎么就攤上這事〕碧荩” “怎么了骗灶?”我有些...
    開封第一講書人閱讀 152,998評論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長秉馏。 經(jīng)常有香客問我耙旦,道長,這世上最難降的妖魔是什么萝究? 我笑而不...
    開封第一講書人閱讀 55,323評論 1 279
  • 正文 為了忘掉前任免都,我火速辦了婚禮,結果婚禮上帆竹,老公的妹妹穿的比我還像新娘绕娘。我一直安慰自己,他們只是感情好栽连,可當我...
    茶點故事閱讀 64,355評論 5 374
  • 文/花漫 我一把揭開白布险领。 她就那樣靜靜地躺著,像睡著了一般秒紧。 火紅的嫁衣襯著肌膚如雪绢陌。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,079評論 1 285
  • 那天熔恢,我揣著相機與錄音下面,去河邊找鬼。 笑死绩聘,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的耗啦。 我是一名探鬼主播凿菩,決...
    沈念sama閱讀 38,389評論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼帜讲!你這毒婦竟也來了衅谷?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 37,019評論 0 259
  • 序言:老撾萬榮一對情侶失蹤似将,失蹤者是張志新(化名)和其女友劉穎获黔,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體在验,經(jīng)...
    沈念sama閱讀 43,519評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡玷氏,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,971評論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了腋舌。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片盏触。...
    茶點故事閱讀 38,100評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出赞辩,到底是詐尸還是另有隱情雌芽,我是刑警寧澤,帶...
    沈念sama閱讀 33,738評論 4 324
  • 正文 年R本政府宣布辨嗽,位于F島的核電站世落,受9級特大地震影響,放射性物質發(fā)生泄漏糟需。R本人自食惡果不足惜屉佳,卻給世界環(huán)境...
    茶點故事閱讀 39,293評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望篮灼。 院中可真熱鬧忘古,春花似錦、人聲如沸诅诱。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,289評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽娘荡。三九已至干旁,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間炮沐,已是汗流浹背争群。 一陣腳步聲響...
    開封第一講書人閱讀 31,517評論 1 262
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留大年,地道東北人换薄。 一個月前我還...
    沈念sama閱讀 45,547評論 2 354
  • 正文 我出身青樓,卻偏偏與公主長得像翔试,于是被迫代替她去往敵國和親轻要。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 42,834評論 2 345

推薦閱讀更多精彩內(nèi)容