View加载绘制流程
- 子线程更新UI异常问题
我们都知道,在子线程进行UI操作是不被允许的,会抛出异常。上周短信的回溯中也是出现了子线程更新UI导致的ANR/闪退问题。
报错示意图如下:
data:image/s3,"s3://crabby-images/b58d3/b58d3474c67dd9dbec2d67213d63ae1c864ee589" alt=""
可以发现其实检查处位于ViewRootImpl里面,具体是checkThread()方法
data:image/s3,"s3://crabby-images/f9d26/f9d26b82e9f4a2450b20cf110bff322437c1bcd6" alt=""
接下来从这里探讨下View的加载过程。
- View 加载过程
从上面的异常栈能够看到,它是一层一层的向上调用requestLayout方法,最终调用到ViewRootImpl里面的方法。这种层级关系就是我们说的view树结构。
以下,我们以activity界面加载为例,探究一下view加载的过程。
我们可以调用setContentView方法来加载自己的布局。同时,安卓的界面是通过window来承载的,所以其实它也被加载到了这个window上,这个过程是怎样的呢?
data:image/s3,"s3://crabby-images/3214d/3214d1694c22ca1edafbe1759fb07fb41b139381" alt=""
其实调用这个方法之前,系统已经创建好了应用的窗口和布局,这里只是将我们自己的view添加进去,让其和activity关联起来。
- performLaunchActivity 方法
activity的onCreate() 、onResume() … 这些生命周期回调,实际上是在ActivityThread中调用的方法。如onCreate就是在performLaunchAcitity中回调。这个方法主要是创建Activity实例、创建窗口,以及建立Actvity和window的关系。
另外在performLaunchAcitity之前还有一大堆的过程,大概就是系统会为我们创建或分配应用进程,并和AMS建立联系,然后再回调到这里。
- activity 的创建
在performLaunchActivity中创建activity的方法如下:
data:image/s3,"s3://crabby-images/c6829/c6829b9eca7ccc3d9615d24c7af2ad0a7c3a2472" alt=""
再进入到红框的代码
data:image/s3,"s3://crabby-images/c93c6/c93c6957d9e3f442461595746aa95c7fdf2d8742" alt=""
可以看到由反射创建activity实例,且用了前面指定的classLoader。
- Attach 的过程
完成activity的创建后,回到performLaunchActivity方法中。接下来会调用attach方法,确定这个activity和window的关系。
data:image/s3,"s3://crabby-images/1c1a5/1c1a55a5b2229599cac9f2a60e2fbdeb23cb8335" alt=""
里面主要创建phoneWindow和获取windowManager:
data:image/s3,"s3://crabby-images/bc14f/bc14fa87e5472200e3fb1319068cfdf45162f45c" alt=""
windowManager通过getSystemService直接获取。
data:image/s3,"s3://crabby-images/3d435/3d435e4baebca7c6660e981fe185d8e13d9a2937" alt=""
到这里创建好了mWindow和mWindowManager。
- onCreate() 回调
再次回到performLaunchActivity(), 从attach方法继续向下,来到callActivityOnCreate()。
data:image/s3,"s3://crabby-images/3f1d0/3f1d0f68127e3aeb783769197345bab1ec0bd0cd" alt=""
这里面会回调onCreate()
- setContentView() 方法
setContentView会调用到mWindow的setContentView方法。也就是2.1.2中创建的PhoneWindow。
data:image/s3,"s3://crabby-images/ef186/ef1866fcf01036a18ace7482cb35b43237c1c383" alt=""
data:image/s3,"s3://crabby-images/1d21f/1d21f04f7108c1c9f9b3e0cdb97f589ea6758da9" alt=""
PhoneWindow的setContentView主要做两件事情:
初始化DecorView,
加载通过参数layoutResID传入的布局文件资源。
data:image/s3,"s3://crabby-images/41b59/41b5937df4f6a885f7e1d1ddd0a3e4ee45e05667" alt=""
data:image/s3,"s3://crabby-images/ac73c/ac73cf0d5889cb58de308b1a43397285ea3be6eb" alt=""
- installDecor()
进入installDecor这个方法主要做两件事:创建Decor和创建ContentParent。
data:image/s3,"s3://crabby-images/1e744/1e74409f364496d6974ac0d10ed35801c1fa34da" alt=""
a. generateDecor()
直接new 一个DecorView,
data:image/s3,"s3://crabby-images/6c8b7/6c8b7f0303072d6e6450eaea8116c2837084d27f" alt=""
DecorView是一个继承自FrameLayout的View。
于是可以得到下图:
data:image/s3,"s3://crabby-images/11f08/11f087d5bd5a1642d8712842577964aef4bc0184" alt=""
b. generateLayout()
之后decorView被当做参数传到generateLayout里面,用于初始化contentParent。
ContentParent 是一个ViewGroup,通过注释我们看到,它会被添加到
DecorView 上。
data:image/s3,"s3://crabby-images/9aba2/9aba2e2b0929b356523cd6cc4308d7e436c595d5" alt=""
而在这个generateLayout中,主要是通过主题做各种对window的配置,从而得到不同的布局。
另外:由于此时处于setContentView中就要用到各种属性,因此设置属性要在setContentView之前才能生效,如下:
data:image/s3,"s3://crabby-images/ffdf6/ffdf618c45fce86feed8e1105205254e2903eaac" alt=""
之后会根据属性选择一个布局,
data:image/s3,"s3://crabby-images/237a5/237a540fce365df32753749692e7a5214edfa46e" alt=""
可以看到其中FrameLayout的id为”@android:id/content”
data:image/s3,"s3://crabby-images/949bc/949bc317e14eb28c093cb2e149ef010918c0e588" alt=""
接下来加载布局到decorView上。
data:image/s3,"s3://crabby-images/85dec/85dec2e076fe59b7eb3b65ecbceb3fd74d72a53b" alt=""
可以看到这里的contentParent就是前面的frameLayout。
之后contentParent创建完成
如下:
data:image/s3,"s3://crabby-images/1a926/1a926f0fb2fb6892cb43ffc1a5b3b3cc9558999c" alt=""
- inflate布局文件
完成了mDecor 和mContentParent 的初始化后,系统创建
的View 树就完成了。之后需要将自身的布局文件,放到这个布局树上。就完成了
布局加载的工作。
直接调用“mLayoutInflater.inflate(layoutResID, mContentParent);”将用户的布
局添加到mContentParent 上。
data:image/s3,"s3://crabby-images/9c7be/9c7be53c7ba50ab67d1def087870777f7c066002" alt=""
注意,inflate是通过反射解析加载view,因此要尽量减少层级嵌套,加快速度。
到此,完成整个布局加载的过程。
- ViewRootImpl 创建过程
到整个View 加载完成,ViewRootImpl 还没有出现。那么接下来看一下ViewRootImpl的创建过程。其实这部分在handleResumeActivity中。ActivityThread 中的handleResumeActivity 主要做两件事,回调onResume方法和更新View。
- performResumeActivity()
首先会执行
data:image/s3,"s3://crabby-images/4b54d/4b54d75c2d79acfd30dcf22f63c15af44e7ca0e1" alt=""
在里面会调用到activity的preformResume方法,并在以下回调onResume。
data:image/s3,"s3://crabby-images/98bc3/98bc3b065b611f73c07281569e0c509bfaa265e7" alt=""
- addView()
回到handleResumeActivity()方法的下面部分。这里会调用wm.addView方法,实际上是在这个方法里创建了ViewRootImpl和完成对View的绘制。
这里会调用addView,实际完成了ViewRootImpl的创建以及View的绘制。
data:image/s3,"s3://crabby-images/36445/3644511d128e71355381947a8299a7a1a88ba072" alt=""
跟进addView方法,它又是通过调用mGlobal的方法实现的。
data:image/s3,"s3://crabby-images/370f3/370f31b152e40ee5d828a82d588601aab46849b2" alt=""
继续进入,可以看到ViewRootImpl被创建。
data:image/s3,"s3://crabby-images/d358b/d358bacb8332095dedea563e2ddcd5d17b3ef03a" alt=""
在ViewRootImpl的构造函数中,会记录当前线程到mThread变量。子线程更新UI报错,检查线程就是检查的这个。
data:image/s3,"s3://crabby-images/f8a2b/f8a2b947538b8bbb1269d2d5d7f4cf15bee4db6c" alt=""
WindowManagerGlobal会将view(DecorView)和root(ViewRootImpl)进行保存。然后调用ViewRootImpl的setView方法。
a. requestLayout()
data:image/s3,"s3://crabby-images/c1341/c1341df4f5bdd781384104e0c9d05217a6927bb7" alt=""
这个过程完成view的测量、布局和绘制。这里会触发view的onLayout()、onMeasure()、onDraw()完成view的绘制即可。详细内容,在下一章节“View更新过程”详细讨论。
b. view.assignParent(this);
data:image/s3,"s3://crabby-images/c5127/c5127905b4f21b7d806315e5023455cf8cd5e9cf" alt=""
会把ViewRootImpl设置为DecorView的parent。
源码阅读到这里,我们增加了WindowManagerImpl、
WindowManagerGlobal 和 ViewRootImpl 三个内容。将这些元素加入到View
结构树中,得到一个比较完整的Activity、Window 和View 之间的关系图。
每个activity 都会有一个phoneWindow,也有
唯一的windowManager 对象。WindowManager 的操作是通过
WindowManagerGlobal 来实现,这是一个单例对象,记录了所有activity 的
decorView、viewRootImpl 信息,通过WindowManagerGlobal 方便对应用窗口
试图进行管理,理论上可以通过它拿到应用所有view 的信息。
data:image/s3,"s3://crabby-images/1b9ef/1b9ef0ec0c07eadc0b1d042ec0286a9a62a80bbf" alt=""
- View 更新过程
WindowManagerImpl.addView() ->
ViewRootImpl.setView()过程中,会调用一次requestLayout()这是第一次对View
树进行自上而下的测量、布局和绘制。注意此时DecorView 还没有将
ViewRootImpl 设为Parent。是由ViewRootImpl 主动发起的刷新。后面的刷新
基本都是由子布局向上请求,再进行自上而下进行刷新。
这部分外面文档讲的很多了,也就是调用顺序scheduleTracersals() -> mTraversalRunnable -> doTraversal() -> performTraversals() -> performMeasure() > onMeasure()、performLayout() > onLayout()、performDraw() > onDraw()。
分为测量, 布局和绘制三部分。
- 子线程更新UI
- 避免requestLayout的向上调用
通过异常调用栈,我们看到在子view中不断向上调用requestLayout,最终调用到viewRootImpl的requestLayout,在其中的checkThread方法中抛出异常。因此只要view更新没有向上调用requestLayout就可以避免这个异常。
data:image/s3,"s3://crabby-images/c42af/c42aff4d9ec265f03543bb6a98909317800b0525" alt=""
调用View的requestLayout方法,会先设置PFLAG_FORCE_LAYOUT标志,这个标志会在layout的时候进行清除。当这个标记存在时,在下面调用mParent.isLayoutRequested()是会返回true,这样就不会再向上调用requestLayout。Android这样设计的目的是避免多次触发重绘。
因此现在主线程先执行requestLayout,然后再在子线程更新UI可以避免抛出异常。
data:image/s3,"s3://crabby-images/19f3d/19f3dcf0301cab74565e607b4dfee91e8b20213b" alt=""
- 在子线程创建ViewRootImpl
data:image/s3,"s3://crabby-images/1f7dc/1f7dcefbf1c606a1ddb10166e84b1dd1ebe3f36e" alt=""
这个mThread是在viewRootImpl创建的时候赋值的,而viewRootImpl是在windowManager.addView里面被创建的。所以如果直接在子线程addView,那之后在这个线程更新view是没问题的。
data:image/s3,"s3://crabby-images/f7590/f7590167dc544bf74aaf87b4068b30d903dc028c" alt=""
- 在onResume之前更新
因为viewRootImpl的创建是在performResumeActivity里面,也就是onResume之后。因此在onResume之前的生命周期里面更新UI,就不会到达ViewRootImpl里面。
但该情况在子线程应避免做耗时操作,否则可能等到viewRootImpl已经被创建并绑定给decorView。