摘要:本文章是基于源碼講解加載顯示基本流程首先上一張圖給大家一個直觀的了解首先一個布局頁面的加載是在中的開始我們就從源碼中的方法入手通過源碼我們可以看到又傳給了中的方法返回的是的對象我們來看中方法是一個隱藏類,在源碼中中方法新創建時為空,調用方
本文章是基于Android源碼6.0講解Activity加載顯示基本流程
首先上一張圖給大家一個直觀的了解
首先一個布局頁面的加載是在Activity中的setContentView(R.layout.res)開始;
我們就從Acitvity源碼中的setContentView方法入手
public void setContentView(@LayoutRes int layoutResID) { getWindow().setContentView(layoutResID); initWindowDecorActionBar(); }
通過源碼我們可以看到layoutResID又傳給了getWindow()中的setContentView(layoutResID)方法;
mWindow = new PhoneWindow(this); public Window getWindow() { return mWindow; }
getWindow()返回的是PhoneWindow的對象
我們來看PhoneWindow中setContentView(layoutResID)方法
PhoneWindow是一個隱藏類,在源碼sources/andorid-23/com/android/internal/policy中
PhoneWindow中setContentView方法
@Override public void setContentView(int layoutResID) { if (mContentParent == null) { installDecor();//Activity新創建時mContentParent 為空,調用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); } mContentParent.requestApplyInsets(); final Callback cb = getCallback(); if (cb != null && !isDestroyed()) { cb.onContentChanged(); } }
我們來看PhoneWindow中installDecor()方法中的關鍵部分
private void installDecor() { if (mDecor == null) { mDecor = generateDecor(); //@1 mDecor.setDescendantFocusability(ViewGroup.FOCUS_AFTER_DESCENDANTS); mDecor.setIsRootNamespace(true); if (!mInvalidatePanelMenuPosted && mInvalidatePanelMenuFeatures != 0) { mDecor.postOnAnimation(mInvalidatePanelMenuRunnable); } } if (mContentParent == null) { mContentParent = generateLayout(mDecor); //@2 ... } mLayoutInflater.inflate(layoutResID, mContentParent);//@3 }
在PhoneWindow源碼中聲明了 private DecorView mDecor;
DecorView 是PhoneWindow中的一個內部類繼承了FrameLayout,是一個幀布局
private final class DecorView extends FrameLayout implements RootViewSurfaceTaker { ... }
@1 首先調用mDecor = generateDecor();返回的是一個 DecorView的對象,這就是所有頁面的跟布局
protected DecorView generateDecor() { return new DecorView(getContext(), -1); }
@2 其次調用mContentParent = generateLayout(mDecor);
protected ViewGroup generateLayout(DecorView decor) { ... } else if ((features & (1 << FEATURE_ACTION_MODE_OVERLAY)) != 0) { layoutResource = R.layout.screen_simple_overlay_action_mode; } else { // Embedded, so no decoration is needed. layoutResource = R.layout.screen_simple; // System.out.println("Simple!"); } 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); ... return contentParent; }
這里的layoutResource是源碼中的布局,我們來看一下R.layout.screen_simple
在sdk/platforms/android-23/data/res/layout路徑中
R.layout.screen_simple就是一個簡單的線性布局,通過mLayoutInflater.inflate和decor.addView添加到DecorView中,
再通過ViewGroup contentParent = (ViewGroup)findViewById(ID_ANDROID_CONTENT);把R.layout.screen_simple中id
為id/content的Framelayout賦值給contentParent
@3 最后通過mLayoutInflater.inflate(layoutResID, mContentParent);把我們寫的布局添加到mContentParent中。
未完待續...
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/71075.html
摘要:此時再次旋轉屏幕時,該不會被系統殺死和重建,只會調用。因此可通過和來判斷是否被重建,并取出數據進行恢復。但需要注意的是,在取出數據時一定要先判斷是否為空。只有在進程不被掉,正常情況下才會執行方法。 目錄介紹 1.0.0.1 說下Activity的生命周期?屏幕旋轉時生命周期?異常條件會調用什么方法? 1.0.0.2 后臺的Activity被系統回收怎么辦?說一下onSaveInsta...
摘要:換句話說,環境或應用程序沒有處于請求操作的適當狀態。項目中異常分析引發崩潰日志的流程分析解決辦法常見的出現場景狀態異常非法線程操作。導致的方法出來顯示消息位于該消息之后,遲遲沒有執行。這時候,的超時檢測結束,刪除了服務中的記錄。 目錄介紹 1.1 java.lang.UnsatisfiedLinkError找不到so庫異常 1.2 java.lang.IllegalStateExce...
閱讀 3606·2021-11-23 09:51
閱讀 1479·2021-11-04 16:08
閱讀 3551·2021-09-02 09:54
閱讀 3619·2019-08-30 15:55
閱讀 2600·2019-08-30 15:54
閱讀 961·2019-08-29 16:30
閱讀 2050·2019-08-29 16:15
閱讀 2320·2019-08-29 14:05