Android进阶 -- Android View绘制流程

本文详细探讨了Android中View的绘制流程,从setContentView开始,解析Activity、Window、DecorView之间的关系,以及如何通过ViewRootImpl调度测量、布局和绘制。文章重点介绍了MeasureSpec的计算方法,以及测量、布局和绘制的细节,对于理解Android UI系统有极大帮助。
摘要由CSDN通过智能技术生成

在日常开发中,相信不少小伙伴都会有这样一个疑问,我们所写的UI控,Android系统是怎么帮我们绘制到屏幕上的呢?

下面将会和大家一起,一步步的去揭开View绘制流程的神秘面纱。

首先,来看setContentView方法,这个方法应该也是用的最多的方法之一,每次在写好Activity之后,都会调用这个方法,将布局文件或者View传进去。在Activity里点击setContentView方法,来到了AppCompatActivity类

    @Override
    public void setContentView(@LayoutRes int layoutResID) {
        getDelegate().setContentView(layoutResID);
    }

    @Override
    public void setContentView(View view) {
        getDelegate().setContentView(view);
    }

    @Override
    public void setContentView(View view, ViewGroup.LayoutParams params) {
        getDelegate().setContentView(view, params);
    }

这里有几个setContentView的重载方法,里面分别都有 getDelegate的setContentView实现,继续看getDelegate的setContentView方法,getDelegate返回的是AppCompatDelegate的实例,而AppCompatDelegate是一个抽象类,返回的是实现类的实例

    //AppCompatDelegate.java
    @NonNull
    public static AppCompatDelegate create(@NonNull Activity activity,
            @Nullable AppCompatCallback callback) {
        return new AppCompatDelegateImpl(activity, callback);
    }
    //AppCompatDelegateImpl.java
    @Override
    public void setContentView(int resId) {
        ensureSubDecor();
        ViewGroup contentParent = mSubDecor.findViewById(android.R.id.content);
        contentParent.removeAllViews();
        LayoutInflater.from(mContext).inflate(resId, contentParent);
        mAppCompatWindowCallback.getWrapped().onContentChanged();
    }

通过这段代码,可以看到,是将resId加载到了contentParent上,而contentParent是android.R.id.content,那么mSubDecor又是什么呢?在ensureSubDecor方法里看到

private void ensureSubDecor() {
    mSubDecor = createSubDecor();
}

接着看createSubDecor

private ViewGroup createSubDecor() {
    ...
    ViewGroup subDecor = null;
    ...
    mWindow.setContentView(subDecor);
    ...
    return subDecor;
}

通过上面的代码,我们可以看出mSubDecor是一个ViewGroup,并且被填充到了mWindow里,而Window作为Activity视图的根窗口,它的setContentView里又做了什么呢?Window同样是一个抽象类,我们来看源码里的唯一实现类PhoneWindow

    @Override
    public void setContentView(View view) {
        setContentView(view, new ViewGroup.LayoutParams(MATCH_PARENT, MATCH_PARENT));
    }    


    @Override
    public void setContentView(View view, ViewGroup.LayoutParams params) {
        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);
        }
        mContentParent.requestApplyInsets();
        final Callback cb = getCallback();
        if (cb != null && !isDestroyed()) {
            cb.onContentChanged();
        }
        mContentParentExplicitlySet = true;
    }

通过上面一段代码,可以看到传进来的view被加载到了mContentParent里,而mContentParent是一个ViewGroup,并且源码里加了详细的注释

    // This is the view in which the window contents are placed. It is either
    // mDecor itself, or a child of mDecor where the contents go.
    ViewGroup mContentParent;

实现代码

// This is the top-level view of the window, containing the window decor.
private DecorView mDecor;

private void installDecor() {
       ...
        if (mContentParent == null) {
            mContentParent = generateLayout(mDecor);
        }
       ...
}

 

看到这里的时候,整个Activity的UI层级关系就比较明朗了,第一层级PhoneWindow(主要和WindowMananger交互), 第二层级DecorView(负责View的测量、定位和绘制),第三层级就是我们加入的ContentView。网络上有很多图片,这里就不贴图了哈

 

    @Override
    public void addView(View child, LayoutParams params) {
        addView(child, -1, params);
    }
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值