Android View的绘制流程
创始人
2024-12-29 08:09:06
0

1.不管是View的添加,还是调用View的刷新方法invalidate()或者requestLayout(),绘制都是从ViewRootImpl的scheduleTraversals()方法开始

void scheduleTraversals() {         if (!mTraversalScheduled) {             mTraversalScheduled = true;             mTraversalBarrier = mHandler.getLooper().getQueue().postSyncBarrier();             mChoreographer.postCallback(                     Choreographer.CALLBACK_TRAVERSAL, mTraversalRunnable, null);             ......         }     } 

首先会发送一个同步屏障消息,以便及时响应UI刷新的异步消息。
然后向Choreographer注册了一个Vsync信号的监听。
2.Vsync信号通过Choreographer回调之后,会执行TraversalRunnable里面的doTraversal()方法

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

doTraversal()方法里首先会移除同步屏障消息,然后执行performTraversals()方法。

performTraversals(){ 	performMeasure(childWidthMeasureSpec, childHeightMeasureSpec); 	 	performLayout(lp, mWidth, mHeight);  	performDraw(mActiveSurfaceSyncGroup) } 

performTraversals()方法里面就会执行我们熟悉的测量、布局、绘制的方法。

3.当测量和布局执行完成后,会执行performDraw()方法,这里分为了开启硬件加速和不开启硬件加速两种绘制方式。

private boolean performDraw(@Nullable SurfaceSyncGroup surfaceSyncGroup) { 	draw(fullRedrawNeeded, surfaceSyncGroup, mSyncBuffer); } private boolean draw(boolean fullRedrawNeeded, @Nullable SurfaceSyncGroup activeSyncGroup,             boolean syncBuffer) { 	if (isHardwareEnabled()) { //如果开启了硬件加速,使用单独的渲染线程 		mAttachInfo.mThreadedRenderer.draw(mView, mAttachInfo, this); 	} else { //使用CPU调用libSkia库渲染 		drawSoftware(surface, mAttachInfo,xOffset,yOffset,scalingRequired, dirty, surfaceInsets) 	} 	 } 

4.开启硬件加速的绘制方式
开启硬件加速,会创建一个RenderThread线程,MainThread会把需要绘制的内容记录到DisplayList里面,并通知RenderThread线程开始渲染,而此时MainThread可以从这个绘制任务中解脱出来。

void draw(View view, AttachInfo attachInfo, DrawCallbacks callbacks) { 	updateRootDisplayList(view, callbacks); //同步View内容到DisplayList 	...... 	syncAndDrawFrame(frameInfo); // 通知渲染线程开始工作 } 

RenderThread会调用libhwui库去做渲染,使用GPU的硬件资源。
不管是否开启硬件加速,只要是绘制,都会先申请空的图形缓冲(dequenebuffer),然后做内容渲染,然后再把有内容的图形缓冲加入消费队列(quenebuffer)。

5.使用软件方式绘制
软件方式绘制渲染,首先通过调用Surface的lockCanvas方法,申请图形缓冲,并关联到Skia图形库;然后调用view的draw方法绘制,这里会调用到View的ondraw()回调方法;绘制完后,通过unlockCanvasAndPost方法把绘制好的缓冲加入到图形缓冲队列,SurfaceFlinger会消费图形缓冲并合成缓冲帧,通过HWC给到屏幕驱动显示。

private boolean drawSoftware(Surface surface, AttachInfo attachInfo, int xoff, int yoff,boolean scalingRequired, Rect dirty, Rect surfaceInsets) { 	canvas = mSurface.lockCanvas(dirty);        	mView.draw(canvas); 	surface.unlockCanvasAndPost(canvas); } 

大概的流程图如下:
在这里插入图片描述

相关内容

热门资讯

专业讨论!德扑线上有机器人,太... 专业讨论!德扑线上有机器人,太坑了从来存在有挂(2020已更新)(哔哩哔哩);详细德扑线上有机器人攻...
带你了解!微扑克线上代打工具,... 带你了解!微扑克线上代打工具,太坑了本然是有挂(2024已更新)(哔哩哔哩);微扑克线上代打工具简单...
六分钟了解!德州ai辅助神器靠... 六分钟了解!德州ai辅助神器靠谱,太坑了起初真的是有挂(2022已更新)(哔哩哔哩);1、超多福利:...
玩家必看科普!德扑数据软件,太... 玩家必看科普!德扑数据软件,太坑了最初是真的有挂(2024已更新)(哔哩哔哩);致您一封信;亲爱德扑...
最新技巧!微扑克网页版辅助,太... 最新技巧!微扑克网页版辅助,太坑了好像真的有挂(2023已更新)(哔哩哔哩);微扑克网页版辅助简单的...
必知教程!德扑之星窥牌,太坑了... 必知教程!德扑之星窥牌,太坑了确实真的是有挂(2026已更新)(哔哩哔哩);1、让任何用户在无需AI...
第三方辅助挂!wopoker用... 第三方辅助挂!wopoker用ai有用,太坑了从前真的有挂(2025已更新)(哔哩哔哩);一、wop...
玩家必知教程!微扑克后台能控制... 玩家必知教程!微扑克后台能控制胜率,太坑了总是真的有挂(2023已更新)(哔哩哔哩);AI辅助机器人...
实测必看!微扑克ai辅助神器,... 实测必看!微扑克ai辅助神器,太坑了原来真的有挂(2024已更新)(哔哩哔哩);1.微扑克ai辅助神...
黑科技免费!德州辅助,太坑了从... 黑科技免费!德州辅助,太坑了从来是真的有挂(2026已更新)(哔哩哔哩)相信很多朋友都在电脑上玩过德...