Activity VS View

  最近有幸作为一面官,有面试了一些新人关于 Android 的基础知识(自己才工作了一年..),或许是处于一点自私的角度,将自己刚毕业时关于 Android 的一点困惑也问了这些新人。果然困惑的人都有同样的困惑,面试他们就像看到了自己刚毕业时候的影子一样。

  本篇主要说明一下 Activity 生命周期和 View 的关联关系,即View 测量(onMeasure())、布局(onLayout())、绘制(onDraw()),以及 View 的一些回调事件比如 onAttachedToWindow()、onDetachedFromWindow()、onFinishInflate()对应Activity的生命周期关系。好多新人对这一块都比较模糊。

  测试代码如下:

/**
 *  MainActivity 布局包含 CustomLayout
 */
public class PtrDemoHomeActivity extends Activity {

    @Override
    protected void onCreate(Bundle bundle) {
        super.onCreate(bundle);
        log("activity onCreate");
        setContentView(R.layout.activity_main);
       // pushFragmentToBackStack(PtrDemoHomeFragment.class, null);
    }

    @Override
    protected void onStart() {
        super.onStart();
        log("activity onStart");
    }

    @Override
    protected void onResume() {
        super.onResume();
        log("activity onResume");
    }

    @Override
    protected void onStop() {
        super.onStop();
        log("activity onStop");
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        log("activity onDestroy");
    }
}

  MainActivity布局文件中,只包含一个简单的自定义的CustomLayout,CustomLayout代码如下:

/**
 * Created by zhangdan on 2018/5/5.
 * <p>
 * comments:
 */
public class CustomLayout extends LinearLayout {

    public CustomLayout(Context context) {
        super(context);
    }

    public CustomLayout(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    @Override
    protected void onAttachedToWindow() {
        super.onAttachedToWindow();
        log("view onAttachedToWindow");
    }

    @Override
    protected void onFinishInflate() {
        super.onFinishInflate();
        log("view onFinishInflate");
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        log("view onMeasure");
    }

    @Override
    protected void onLayout(boolean changed, int l, int t, int r, int b) {
        super.onLayout(changed, l, t, r, b);
        log("view onLayout");
    }

    @Override
    protected void onDetachedFromWindow() {
        super.onDetachedFromWindow();
        log("on DetachedFromWindow");
    }
}

运行结果如下:


result

下面从代码角度依次分析一下上面执行顺序的原因。

Activity.onCreate() > View.onFinishInflate() :

当需要启动一个Activity时候,最后都会由AMS调用到Activity Thread的performLaunchActivity(),在该方法中,首先会回调到Activity的onCreate()方法,平时我们都是在onCreate中setContentView()去设置界面布局,那么setContentView()和View.onFinishInflate()有什么关系呢,代码注释如下:

Step 1 :   PhoneWindow.setContentView();
// Activity 的setContentView() 最终会调用到 PhoneWindow 的 setContentView()
// 在 PhoneWindow 中开始真正加载布局
public void setContentView(int layoutResID) {
        if (mContentParent == null) {
            installDecor();
        } else if (!hasFeature(FEATURE_CONTENT_TRANSITIONS)) {
            mContentParent.removeAllViews();
        }

        // setContentView() 最终会通过LayoutInflater 去加载布局
        mLayoutInflater.inflate(layoutResID, mContentParent);
        mContentParent.requestApplyInsets();
        final Callback cb = getCallback();
        if (cb != null && !isDestroyed()) {
            cb.onContentChanged();
        }
        ......
    }

Step 2 :   LayoutInflater.inflate();  // 解析给定的 XML 布局文件

 public View inflate(XmlPullParser parser, @Nullable ViewGroup root, boolean attachToRoot) {
       .....
           ....
                    // Inflate all children under temp against its context.
                    //  注:在此处开始 inflateChild(),注: 最后一个参数为
                    // true , 代表是否接受 finishInflate() 回调。
                    rInflateChildren(parser, temp, attrs, true);

                    if (DEBUG) {
                        System.out.println("-----> done inflating children");
                    }

                    // We are supposed to attach all the views we found (int temp)
                    // to root. Do that now.
                    if (root != null && attachToRoot) {
                        root.addView(temp, params);
                    }

                    // Decide whether to return the root that was passed in or the
                    // top view found in xml.
                    if (root == null || !attachToRoot) {
                        result = temp;
                    }
                }
        ......    
    }

Step 3 :   LayoutInflater.rInflate();  // 递归解析布局

final int depth = parser.getDepth();
        int type;

        while (((type = parser.next()) != XmlPullParser.END_TAG ||
                parser.getDepth() > depth) && type != XmlPullParser.END_DOCUMENT) {
                ...........
                // 递归调用,直到布局解析完成为止。
                rInflateChildren(parser, view, attrs, true);
                viewGroup.addView(view, params);
            }
        }

        if (finishInflate) {
            // 注 : 此处 view 接收 onFinishInflate() 回调。
            parent.onFinishInflate();
        }

综上所述,在Activity setContentView()中,最终会调用到 LayoutInflate 中的 reInflate() 方法去递归解析布局,在布局解析完成以后,会调用 View.onFinishInflate() 回调。因此 View.onFinishInflate()调用时机在 setCotentView() 之后会立即调用到。所以如果有自定义 View 的话, findViewById() 可以放到该回调里面去做。

Activity.onResume() > View.onAttachedToWindow() :

在 ActivityThread 的 handleLaunchActivity() 中,Activity 的生命周期会依次得到回调,代码如下:

private void handleLaunchActivity(ActivityClientRecord r, Intent customIntent, String reason) {
        
        注 : 在此处会执行 Activity onCreate() 、onStart() 回调。
        Activity a = performLaunchActivity(r, customIntent);

        if (a != null) {
            r.createdConfig = new Configuration(mConfiguration);
            reportSizeConfigurations(r);
            Bundle oldState = r.state;

            注 : 在此处会执行到 Activity onResume() 回调。
            handleResumeActivity(r.token, false, r.isForward,
                    !r.activity.mFinished && !r.startsNotResumed, r.lastProcessedSeq, reason);
           .......
           }
}

从上述代码可以看出,从一个 Activity 的启动开始 handleLaunchActivity(),便依次执行了 onCreate()、onStart()、onResume() , 那么Activity 中 View 的测量绘制相关操作是在哪一步完成的呢。经常有听人回答到,在 onResume() 中,界面就真正可见可交互。但是这种方法是大错特错的,因为在 Activity onResume()中,View相关的测量绘制并没有开始! view 测量绘制真正开始时机是在 View.onAttachedToWindow()之后的。那么 onAttachedToWindow() 是什么时候调用的呢,看下一小节分析。

View.onAttachedToWindow() > View.onMeasure() :

在上一小节,可以看到在 Activity handleLaunchActivity() 中,最后会调用到 handleResumeActivity(),那么handleResumeActivity() 除了最后调用了 Activity onResume() 以后,具体还做了什么工作呢,代码如下:

final void handleResumeActivity(IBinder token,
                                    boolean clearHide, boolean isForward, boolean reallyResume, int seq, String reason) {
        .......
        // TODO Push resumeArgs into the activity for consideration
        //  Step 1 :  注 : 该方法会执行 Activity onResume() 回调。
        r = performResumeActivity(token, clearHide, reason);

        if (r != null) {
            final Activity a = r.activity;

            // If the window hasn't yet been added to the window manager,
            // and this guy didn't finish itself or start another activity,
            // then go ahead and add the window.
            boolean willBeVisible = !a.mStartedActivity;
            if (!willBeVisible) {
                try {
                    willBeVisible = ActivityManagerNative.getDefault().willActivityBeVisible(
                            a.getActivityToken());
                } catch (RemoteException e) {
                    throw e.rethrowFromSystemServer();
                }
            }

            // Step 2 :  注 : 调用 WindowManager.addView() 将 Activity 
            对应的 window 添加到 WMS 中。
            if (r.window == null && !a.mFinished && willBeVisible) {
                r.window = r.activity.getWindow();
                View decor = r.window.getDecorView();
                decor.setVisibility(View.INVISIBLE);
                ViewManager wm = a.getWindowManager();
                WindowManager.LayoutParams l = r.window.getAttributes();
                a.mDecor = decor;
                l.type = WindowManager.LayoutParams.TYPE_BASE_APPLICATION;
                l.softInputMode |= forwardBit;
                
                if (a.mVisibleFromClient && !a.mWindowAdded) {
                    a.mWindowAdded = true;
            // Step 3 :  注 : 调用 WindowManager.addView() 将 Activity 
            对应的 window 添加到 WMS 中。 decor 为 Activity 的根布局
                    wm.addView(decor, l);
                }
            } 
        } 

在 handleResumeActivity() 中,最终会调用到 WindowManager.addView() 去往 WSM 中添加 window ,最终该方法会 new 一个 ViewRootImpl 实例,并调用 requestLayout() 去请求布局,代码如下 :

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

void scheduleTraversals() {
        if (!mTraversalScheduled) {
            mTraversalScheduled = true;
            mTraversalBarrier = mHandler.getLooper().getQueue().postSyncBarrier();
            // 将 traversalRunnable 放到 messageQueue 中,准备执行
            mChoreographer.postCallback(
                    Choreographer.CALLBACK_TRAVERSAL, mTraversalRunnable, null);
            if (!mUnbufferedInputDispatch) {
                scheduleConsumeBatchedInput();
            }
            notifyRendererOfFramePending();
            pokeDrawLockIfNeeded();
        }
    }

在 viewRootImpl 执行 requestLayout() 的时候,会将mTraversalRunnable 以消息的形式存放于主线程的 MessageQueue中,mTraversalRunnable 中包含真正的view 测量、绘制等逻辑,mTraversalRunnable 的代码如下:


final class TraversalRunnable implements Runnable {
        @Override
        public void run() {
            doTraversal();
        }
    }

void doTraversal() {
        if (mTraversalScheduled) {
            mTraversalScheduled = false;
            mHandler.getLooper().getQueue().removeSyncBarrier(mTraversalBarrier);

            if (mProfile) {
                Debug.startMethodTracing("ViewAncestor");
            }
            // 在 performTraversals 中,开始真正执行 view 相关操作
            performTraversals();

            if (mProfile) {
                Debug.stopMethodTracing();
                mProfile = false;
            }
        }
    }

private void performTraversals() {
       ..... 
       // 注: host 为 decorView , 会依次分发,最后回调
       // 到 view onAttachedToWindow()
       host.dispatchAttachedToWindow(mAttachInfo, 0);
       .....
       // 测量
       performMeasure();
       .....
       // 布局
       performLayout();
       ......
       // 绘制
       performDraw();
       .....
}

综上可以看出,在 Activity onResume() 完成之后,会依次执行 view onAttachToWindow() , onMeasure()、onLayout()等。所以在 onResume() 中,并拿不到控件的宽高,因为此时,view 的测量根本都还没有开始!

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

推荐阅读更多精彩内容