个人笔记---Window,WindowManager和ViewRoot之间不可描述的关系

前言

引用一下刚哥关于Window、WindowManager和RootViewImpl的概念

Window表示一个窗口的概念,是一个抽象类,它的具体实现是PhoneWindow,而创建一个Window,只需要通过WindowManager就可完成,WindowManager是外界访问Window的入口,Window的具体实现在WindowMangerService中,WindowManager和WindowMangerService的交互实际上是一个IPC的过程

android中所有的视图都是通过Window来呈现的,不管是Activity,Dialog或Toast,它们的视图实际上都是附加在Window上面的,因此Window是View的直接管理者,从View的事件分发机制可知,单击事件由Window传递到DecorView,然后再由DecorView传递给View,就连Activity的设置视图的方法setContentView底层也是通过Window完成的

ViewRoot对应于ViewRootImpl,它是连接WindowManager和DecorView之间的纽带,View的measure、layout和draw过程都是由它来完成的

View的事件分发机制传送门

什么时候创建它们

它们的创建要从Activity的启动开始说起,而Activity的启动是一个相当复杂的过程,这篇文章的重点不在这里,所有不会详细分析Activity的启动过程。
当启动一个Activity的时候,我们只需要调用startActivity方法,然后系统会调用一系列startActivity的方法,最终会调用ActivityThread里面的performLaunchActivity方法

try {
            //首先通过Instrumentation的newActivity方法使用类加载器创建activity对象
            java.lang.ClassLoader cl = appContext . getClassLoader ();
            activity = mInstrumentation.newActivity(
                cl, component.getClassName(), r.intent);
            StrictMode.incrementExpectedActivityCount(activity.getClass());
            r.intent.setExtrasClassLoader(cl);
            r.intent.prepareToEnterProcess();
            if (r.state != null) {
                r.state.setClassLoader(cl);
            }
        } catch (Exception e) {
            if (!mInstrumentation.onException(activity, e)) {
                throw new RuntimeException (
                        "Unable to instantiate activity " + component
                                + ": " + e.toString(), e);
            }
        }
...
//通过activity的attach方法,传入window,创建window
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, r.configCallback);

继续看一下Activity里面的attach方法,看一下是否创建了Window对象

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, ActivityConfigCallback activityConfigCallback) {
        ...
        //创建window对象
        mWindow = new PhoneWindow(this, window, activityConfigCallback);
        mWindow.setWindowControllerCallback(this);
        mWindow.setCallback(this);
        mWindow.setOnWindowDismissedCallback(this);
        mWindow.getLayoutInflater().setPrivateFactory(this);
        if (info.softInputMode != WindowManager.LayoutParams.SOFT_INPUT_STATE_UNSPECIFIED) {
            mWindow.setSoftInputMode(info.softInputMode);
        }
        if (info.uiOptions != 0) {
            mWindow.setUiOptions(info.uiOptions);
        }
        ...
        //设置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());
        }
        mWindowManager = mWindow.getWindowManager();
        mCurrentConfig = config;

        mWindow.setColorMode(info.colorMode);

        setAutofillCompatibilityEnabled(application.isAutofillCompatibilityEnabled());
        enableAutofillCompatibilityIfNeeded();
    }

上面的代码很清晰,通过实例创建window对象,通过context. getSystemService获取WindowManager。

此时window和WindowManager已经创建好,Instrumentation会调用callActivityOnCreate方法,

Instrumentation:
public void callActivityOnCreate(Activity activity, Bundle icicle) {
        prePerformCreate(activity);
        activity.performCreate(icicle);
        postPerformCreate(activity);
    }

Activity:
final void performCreate(Bundle icicle, PersistableBundle persistentState) {
        mCanEnterPictureInPicture = true;
        restoreHasCurrentPermissionRequest(icicle);
        //onCreate方法被调用
        if (persistentState != null) {
            onCreate(icicle, persistentState);
        } else {
            onCreate(icicle);
        }
        writeEventLog(LOG_AM_ON_CREATE_CALLED, "performCreate");
        mActivityTransitionState.readState(icicle);

        mVisibleFromClient = !mWindow.getWindowStyle().getBoolean(
                com.android.internal.R.styleable.Window_windowNoDisplay, false);
        mFragments.dispatchActivityCreated();
        mActivityTransitionState.setEnterActivityOptions(this, getActivityOptions());
    }

Activity的onCreate方法被调用,在这个方法中,我们一般会通过setContentView(int layoutId)设置一个contentView

Activity:
public void setContentView(@LayoutRes int layoutResID) {
        //通过Window设置contentView
        getWindow().setContentView(layoutResID);
        initWindowDecorActionBar();
    }

PhoneWindow:
public void setContentView(int layoutResID) {
        if (mContentParent == null) {
            installDecor();
        } else if (!hasFeature(FEATURE_CONTENT_TRANSITIONS)) {
            mContentParent.removeAllViews();
        }
    }

PhoneWindow通过installDecor()方法创建DecorView,此时DecorView也已经就绪,还剩下ViewRoot没有创建。那么它到底是什么时候创建的。这个时候我们继续跟踪ActivityThread的代码,在handleResumeActivity中有这样一句代码

ActivityThread:
public void handleResumeActivity(IBinder token, boolean finalStateRequest, boolean isForward, String reason) {
    ...
    r.activity.makeVisible();
}

Activity:
void makeVisible() {
   if (!mWindowAdded) {
      ViewManager wm = getWindowManager();
      //此时WindowManager将Window添加进来
      wm.addView(mDecor, getWindow().getAttributes());
      mWindowAdded = true;
   }
   mDecor.setVisibility(View.VISIBLE);
}

WindowManager是一个接口,它的实现类是WindowManagerImpl,看一下实现类里面addView的代码

public void addView(@NonNull View view, @NonNull ViewGroup.LayoutParams params) {
    applyDefaultToken(params);
    mGlobal.addView(view, params, mContext.getDisplay(), mParentWindow);
}

可以看出WindowManagerImpl并没有实现addView的操作,而是交给了WindowManagerGlobal来处理,WindowManagerGlobal的addView方法主要分为如下几步,

1.检查参数是否合法,如果是子view还需要调整一些布局参数

if (view == null) {
    throw new IllegalArgumentException ("view must not be null");
}
if (display == null) {
    throw new IllegalArgumentException ("display must not be null");
}
if (!(params instanceof WindowManager.LayoutParams)) {
    throw new IllegalArgumentException ("Params must be WindowManager.LayoutParams");
}

final WindowManager . LayoutParams wparams = (WindowManager.LayoutParams) params;
if (parentWindow != null) {
    parentWindow.adjustLayoutParamsForSubWindow(wparams);
} 

2.创建ViewRootImpl,将一些对象添加到对应的列表中

//创建ViewRootImpl实例
root = new ViewRootImpl (view.getContext(), display);

view.setLayoutParams(wparams);
//将view、root和params添加到对应的列表中
mViews.add(view);
mRoots.add(root);
mParams.add(wparams);

在WindowManagerGlobal中维护了几个比较重要的列表

//存储所有Window所对应的View
private final ArrayList<View> mViews = new ArrayList<View>();
//存储所有Window所对应的ViewRootImpl
private final ArrayList<ViewRootImpl> mRoots = new ArrayList<ViewRootImpl>();
//存储所有Window所对应的布局参数
private final ArrayList<WindowManager.LayoutParams> mParams = new ArrayList<WindowManager.LayoutParams>();
//存储那些正在被删除的View对象,或者说那些已经调用了removeView方法但删除操作还未完成的Window对象
private final ArraySet<View> mDyingViews = new ArraySet<View>();

3.通过ViewRootImpl更新界面并完成Window的添加过程

public void setView(View view, WindowManager.LayoutParams attrs, View panelParentView) {
    ...
    requestLayout();
    try {
        mOrigWindowType = mWindowAttributes.type;
        mAttachInfo.mRecomputeGlobalAttributes = true;
        collectViewAttributes();
        res = mWindowSession.addToDisplay(
            mWindow, mSeq, mWindowAttributes,
            getHostVisibility(), mDisplay.getDisplayId(), mWinFrame,
            mAttachInfo.mContentInsets, mAttachInfo.mStableInsets,
            mAttachInfo.mOutsets, mAttachInfo.mDisplayCutout, mInputChannel);
    }
    ...
}

public void requestLayout() {
    if (!mHandlingLayoutInLayoutRequest) {
        checkThread();
        mLayoutRequested = true;
        scheduleTraversals();
    }
}

这里我们只看一下核心代码,ViewRootImpl通过requestLayout来完成异步刷新请求,scheduleTraversals()则是View绘制的入口,View绘制的内容也不多说,后续会单独介绍。接着会通过WindowSession完成最终的添加过程,mWindowSession的类型是IWindowSession,是一个Binder对象,真正的实现类是Session,到此Window的添加完成了一次IPC调用

public int addToDisplay(IWindow window, int seq, WindowManager.LayoutParams attrs,
    int viewVisibility, int displayId, Rect outFrame, Rect outContentInsets,
    Rect outStableInsets, Rect outOutsets,
    DisplayCutout.ParcelableWrapper outDisplayCutout, InputChannel outInputChannel) {
    return mService.addWindow(
        this, window, seq, attrs, viewVisibility, displayId, outFrame,
        outContentInsets, outStableInsets, outOutsets, outDisplayCutout, outInputChannel);
}

而Session的addToDisplay又调用了WindowManagerService的addWindow来实现添加,至此,Window的添加过程分析完毕,删除过程同添加过程一样,这里就不做分析了。

总结

1. 一个Activity对应一个Window
2. Window是一个抽象的概念,相当于一个容器,内部持有一个DecorView,是一个根view,唯一实现类是PhoneWindow
3. ViewRootImpl完成View的绘制过程
4. ViewRootImpl是View和WindowManager之间的桥梁,View通过WindowManager间接调用ViewRootImpl的方法, ViewRootImpl通过WindowSession单向与WindowManagerService通信,最终通过WindowManagerService完成对View的添加;
5. 通过WindowSession添加view时,传入了一个IWindow的对象,它也是一个Binder对象,WindowManagerService通过IWindow与ViewRootImpl进行单向通信


个人能力有限,如有错误,欢迎指正

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,980评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,178评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,868评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,498评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,492评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,521评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,910评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,569评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,793评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,559评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,639评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,342评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,931评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,904评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,144评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,833评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,350评论 2 342

推荐阅读更多精彩内容