View的工作流程,就是View进行measurelayoutdraw的过程,本篇文章我们就来一起看一下View是如何开始他的工作流程的。

(注:文中源码基于 Android 12

在上篇文章《View体系(五)熟悉又陌生的setContentView》中我们讲过Activity的结构,包括了ActivityPhoneWindowDecorView。讲了PhoneWindow的创建和DecorView的创建,但此时DecorView还没有加载到PhoneWindow中,下面我们就从源码看一下DecorView是如何加载到PhoneWindow中的。

当我们调用ActivitystartActivity时,最终会调用到ActivityThreadhandleLaunchActivity,代码如下:

android.app.ActivityThread

1
2
3
4
5
public Activity handleLaunchActivity(ActivityClientRecord r,
PendingTransactionActions pendingActions, Intent customIntent) {
......
final Activity a = performLaunchActivity(r, customIntent);//1
......

注释1处调用performLaunchActivity创建了一个Activity,进入performLaunchActivity方法:

android.app.ActivityThread

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
private Activity performLaunchActivity(ActivityClientRecord r, Intent customIntent) {
...
Activity activity = null;
...
activity = mInstrumentation.newActivity(
cl, component.getClassName(), r.intent);//1
...
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,
r.assistToken, r.shareableActivityToken);//2
...
mInstrumentation.callActivityOnCreate(activity, r.state, r.persistentState);//3
...
return activity;

注释1处会通过mInstrumentationnewActivity方法,使用反射的方式,创建Activity的实例,注释2处调用activity.attach方法,在之前的文章中讲过,attach会创建PhoneWindow实例,并赋值给传入的window引用,注释3处最终会调用ActivityOnCreate回调方法,进而调用setContentView方法,创建DecorView

Android12的源码中,handleLaunchActivity方法中并不会直接调用handleResumeActivity方法,而是通过ClientTransaction类来实现调用handleLaunchActivity之后再调用handleResumeActivity,具体的放在后面的文章中讲解。

我们继续看handleResumeActivity方法:

android.app.ActivityThread

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
public void handleResumeActivity(ActivityClientRecord r, boolean finalStateRequest,
boolean isForward, String reason) {
...
if (r.window == null && !a.mFinished && willBeVisible) {
r.window = r.activity.getWindow();
View decor = r.window.getDecorView();//1
decor.setVisibility(View.INVISIBLE);
ViewManager wm = a.getWindowManager();//2
WindowManager.LayoutParams l = r.window.getAttributes();
a.mDecor = decor;
l.type = WindowManager.LayoutParams.TYPE_BASE_APPLICATION;
l.softInputMode |= forwardBit;
if (r.mPreserveWindow) {
a.mWindowAdded = true;
r.mPreserveWindow = false;
ViewRootImpl impl = decor.getViewRootImpl();
if (impl != null) {
impl.notifyChildRebuilt();
}
}
if (a.mVisibleFromClient) {
if (!a.mWindowAdded) {
a.mWindowAdded = true;
wm.addView(decor, l);//3
...

注释1处得到了DecorView,注释2处得到了WindowManager对象,注释3处调用WindowManageraddView方法,将DecorView作为参数传入,WindowManager的实现类是WindowManagerImpl,所以实际调用的是WindowManagerImpladdView方法。再看WindowManagerImpladdView方法:

android.view.WindowManagerImpl

1
2
3
4
5
public void addView(@NonNull View view, @NonNull ViewGroup.LayoutParams params) {
applyTokens(params);
mGlobal.addView(view, params, mContext.getDisplayNoVerify(), mParentWindow,
mContext.getUserId());//1
}

注释1处又调用了WindowManagerGlobaladdView方法:

android.view.WindowManagerGlobal

1
2
3
4
5
6
7
8
public void addView(View view, ViewGroup.LayoutParams params,
Display display, Window parentWindow, int userId) {
...
ViewRootImpl root;
...
root = new ViewRootImpl(view.getContext(), display);//1
...
root.setView(view, wparams, panelParentView, userId);//2

注释1处创建了ViewRootImpl实例赋值给root,注释2处将DecorView作为参数传入ViewRootImpl,完成了DecorViewWindow的绑定。

ViewRootImpl中还有个方法performTraversals,这个方法使View进入正真的工作流程:

android.view.ViewRootImpl

1
2
3
4
5
6
7
8
private void performTraversals() {
...
performMeasure(childWidthMeasureSpec, childHeightMeasureSpec);//1
...
performLayout(lp, mWidth, mHeight);//2
...
performDraw();//3
...

注释1处会执行viewMeasure过程,注释2处会执行viewLayout过程,注释3处会执行viewDraw过程。

关注我