Android学习札记15:对Android中View绘制流程的一些理解
Android学习札记15:对Android中View绘制流程的一些理解
分类:Android 2012-06-27 14:36 284人阅读 评论(0) 收藏 举报 android layout hierarchy matrix null traversal整个View树的绘制流程是在ViewRoot.java类中的performTraversals()方法展开的,该函数的执行过程可简单概况为:根据之前设置的状态,判断是否需要重新计算视图大小(measure)、是否需要重新安置视图的位置(layout)、以及是否需要重新绘制(draw)。
流程一:mesarue()测量过程
主要作用:为整个View树计算实际的大小,即设置实际的高(对应属性:mMeasuredHeight)和宽(对应属性:mMeasureWidth),每个View控件的实际高宽都是由父视图和本身视图决定的。
具体的调用过程:ViewRoot的属性mView(其类型一般为ViewGroup类型)调用measure()方法去计算View树的大小,回调View/ViewGroup对象的onMeasure()方法,该方法实现的功能如下:
1、设置本View视图的最终大小,该功能的实现通过调用setMeasuredDimension()方法去设置实际的高(对应属性:mMeasuredHeight)和宽(对应属性:mMeasureWidth);
2、如果该View对象是ViewGroup类型,需要重写onMeasure()方法,对其子视图进行遍历的measure()过程。对每个子视图的measure()过程,是通过调用父类ViewGroup.java类里的measureChildWithMargins()方法去实现,该方法内部只是简单地调用了View对象的measure()方法。(由于measureChildWithMargins()方法只是一个过渡,更简单的做法是直接调用View对象的measure()方法)。
整个measrue()的调用流程就是个树形的递归过程。
measure()方法的源代码:
[java] view plain copy
- Thisiscalledtofindouthowbigaviewshouldbe.Theparentsuppliesconstraintinformationinthewidthandheightparameters.
- TheactualmesurementworkofaviewisperformedinonMeasure(int,int),calledbythismethod.Therefore,onlyonMeasure(int,int)canandmustbe
- overridenbysubclasses.
- Parameters:
- widthMeasureSpecHorizontalspacerequirementsasimposedbytheparent
- heightMeasureSpecVerticalspacerequirementsasimposedbytheparent
- Seealso:
- onMeasure(int,int)
- publicfinalvoidmeasure(intwidthMeasureSpec,intheightMeasureSpec){
- if((mPrivateFlags&FORCE_LAYOUT)==FORCE_LAYOUT||
- widthMeasureSpec!=mOldWidthMeasureSpec||
- heightMeasureSpec!=mOldHeightMeasureSpec){
- //firstclearsthemeasureddimensionflag
- mPrivateFlags&=~MEASURED_DIMENSION_SET;
- if(ViewDebug.TRACE_HIERARCHY){
- ViewDebug.trace(this,ViewDebug.HierarchyTraceType.ON_MEASURE);
- }
- //measureourselves,thisshouldsetthemeasureddimensionflagback
- onMeasure(widthMeasureSpec,heightMeasureSpec);
- //flagnotset,setMeasuredDimension()wasnotinvoked,weraise
- //anexceptiontowarnthedeveloper
- if((mPrivateFlags&MEASURED_DIMENSION_SET)!=MEASURED_DIMENSION_SET){
- thrownewIllegalStateException("onMeasure()didnotsetthe"
- +"measureddimensionbycalling"
- +"setMeasuredDimension()");
- }
- mPrivateFlags|=LAYOUT_REQUIRED;
- }
- mOldWidthMeasureSpec=widthMeasureSpec;
- mOldHeightMeasureSpec=heightMeasureSpec;
- }
为了大家更好的理解,采用“二B程序员”的方式利用伪代码描述该measure流程
[java] view plain copy
- //ViewRoot.java
- //measure()过程
- //发起measure()过程的"发号者"是在ViewRoot.java里的performTraversals()方法中的mView.measure()
- privatevoidperformTraversals(){
- //...
- ViewmView;
- mView.measure(h,l);
- //....
- }
- //回调View视图里的onMeasure过程
- privatevoidonMeasure(intheight,intwidth){
- //设置该view的实际高(mMeasuredHeight)和宽(mMeasuredWidth)
- //1、该方法必须在onMeasure调用,否者报异常。
- setMeasuredDimension(h,l);
- //2、如果该View是ViewGroup类型,则对它的每个子View进行measure()过程
- intchildCount=getChildCount();
- for(inti=0;i<childCount;i++){
- //2.1、获得每个子View对象引用
- Viewchild=getChildAt(i);
- //整个measure()过程就是个递归过程
- //该方法只是一个过滤器,最后会调用measure()过程或者measureChild(child,h,i)方法
- measureChildWithMargins(child,h,i);
- //其实,对于我们自己写的应用来说,最好的办法是去掉框架里的该方法,直接调用view.measure(),如下:
- //child.measure(h,l);
- }
- }
- //该方法在ViewGroup.java里具体实现
- protectedvoidmeasureChildWithMargins(Viewv,intheight,intwidth){
- v.measure(h,l);
- }
流程二:layout()布局过程
主要作用:根据子视图的大小以及布局参数将View树放在合适的位置上。
具体的调用过程:host.layout()开始View树的布局,继而回调给View/ViewGroup类中的layout()方法,具体过程如下:
1、layout()方法会设置该View视图位于父视图的坐标轴,即mLeft、mTop、mLeft、mBottom(调用setFrame()函数去实现),接下来回调onLayout()方法(如果该View是ViewGroup对象,需要实现该方法,对每个子视图进行布局);
2、如果该View是ViewGroup类型,需要遍历每个子视图childView,调用该子视图的layout()方法去设置它的坐标值。
layout()方法的源代码:
[java] view plain copy
- Assignasizeandpositiontoaviewandallofitsdescendants
- Thisisthesecondphaseofthelayoutmechanism.(Thefirstismeasuring).Inthisphase,eachparentcallslayoutonallofitschildrentopositionthem.
- Thisistypicallydoneusingthechildmeasurementsthatwerestoredinthemeasurepass().
- Derivedclassesshouldnotoverridethismethod.DerivedclasseswithchildrenshouldoverrideonLayout.Inthatmethod,theyshouldcalllayoutoneach
- oftheirchildren.
- Parameters:
- lLeftposition,relativetoparent
- tTopposition,relativetoparent
- rRightposition,relativetoparent
- bBottomposition,relativetoparent
- @SuppressWarnings({"unchecked"})
- publicvoidlayout(intl,intt,intr,intb){
- intoldL=mLeft;
- intoldT=mTop;
- intoldB=mBottom;
- intoldR=mRight;
- booleanchanged=setFrame(l,t,r,b);//设置每个视图位于父视图的位置坐标
- if(changed||(mPrivateFlags&LAYOUT_REQUIRED)==LAYOUT_REQUIRED){
- if(ViewDebug.TRACE_HIERARCHY){
- ViewDebug.trace(this,ViewDebug.HierarchyTraceType.ON_LAYOUT);
- }
- onLayout(changed,l,t,r,b);//回调onLayout()方法,设置每个子视图的布局
- mPrivateFlags&=~LAYOUT_REQUIRED;
- ListenerInfoli=mListenerInfo;
- if(li!=null&&li.mOnLayoutChangeListeners!=null){
- ArrayList<OnLayoutChangeListener>listenersCopy=
- (ArrayList<OnLayoutChangeListener>)li.mOnLayoutChangeListeners.clone();
- intnumListeners=listenersCopy.size();
- for(inti=0;i<numListeners;++i){
- listenersCopy.get(i).onLayoutChange(this,l,t,r,b,oldL,oldT,oldR,oldB);
- }
- }
- }
- mPrivateFlags&=~FORCE_LAYOUT;
- }
同样地,将上面layout()调用流程用伪代码描述如下:
[java] view plain copy
- //ViewRoot.java
- //layout()过程
- //发起layout()过程的“发号者”是在ViewRoot.java里的performTraversals()方法中的mView.layout()
- privatevoidperformTraversals(){
- //...
- ViewmView;
- mView.layout(left,top,right,bottom);
- //....
- }
- //回调View视图里的onLayout过程,该方法只由ViewGroup类型实现
- privatevoidonLayout(intleft,inttop,right,bottom){
- //如果该View不是ViewGroup类型
- //调用setFrame()方法设置该控件的在父视图上的坐标轴
- setFrame(l,t,r,b);
- //--------------------------
- //如果该View是ViewGroup类型,则对它的每个子View进行layout()过程
- intchildCount=getChildCount();
- for(inti=0;i<childCount;i++){
- //获得每个子View对象引用
- Viewchild=getChildAt(i);
- //整个layout()过程就是个递归过程
- child.layout(l,t,r,b);
- }
- }
流程三:draw()绘制过程
由ViewRoot对象的 performTraversals()方法调用draw()方法发起绘制该View树的过程,值得注意的是每次发起绘图时,并不会重新绘制每个View树的视图,而只会重新绘制那些“需要重绘”的视图,View类内部变量包含了一个标志位DRAWN,当该视图需要重绘时,就会为该View添加该标志位。
具体的调用过程:mView.draw()开始绘制,draw()方法实现的功能如下:
1、绘制该View的背景;
2、为显示渐变框做一些准备操作(见5,大多数情况下,不需要改渐变框);
3、调用onDraw()方法绘制视图本身(每个View都需要重载该方法,ViewGroup不需要实现该方法);
4、dispatchDraw()方法内部会遍历每个子视图,调用drawChild()去重新回调每个子视图的draw()方法(注意,这个地方“需要重绘”的视图才会调用draw()方法)。如果该View类型不为ViewGroup,即不包含子视图,不需要重载该方法。值得说明的是,ViewGroup类已经为我们重写了dispatchDraw()的功能实现, 应用程序一般不需要重写该方法,但可以重载父类函数实现具体的功能;
5、绘制滚动条。
于是,整个调用过程就这样递归下去了。
draw()方法的源代码:
[java] view plain copy
- /**
- *Manuallyrenderthisview(andallofitschildren)tothegivenCanvas.
- *Theviewmusthavealreadydoneafulllayoutbeforethisfunctionis
- *called.Whenimplementingaview,implement
- *{@link#onDraw(android.graphics.Canvas)}insteadofoverridingthismethod.
- *Ifyoudoneedtooverridethismethod,callthesuperclassversion.
- *
- *@paramcanvasTheCanvastowhichtheViewisrendered.
- */
- publicvoiddraw(Canvascanvas){
- if(ViewDebug.TRACE_HIERARCHY){
- ViewDebug.trace(this,ViewDebug.HierarchyTraceType.DRAW);
- }
- finalintprivateFlags=mPrivateFlags;
- finalbooleandirtyOpaque=(privateFlags&DIRTY_MASK)==DIRTY_OPAQUE&&
- (mAttachInfo==null||!mAttachInfo.mIgnoreDirtyState);
- mPrivateFlags=(privateFlags&~DIRTY_MASK)|DRAWN;
- /*
- *Drawtraversalperformsseveraldrawingstepswhichmustbeexecuted
- *intheappropriateorder:
- *
- *1.Drawthebackground
- *2.Ifnecessary,savethecanvas'layerstoprepareforfading
- *3.Drawview'scontent
- *4.Drawchildren
- *5.Ifnecessary,drawthefadingedgesandrestorelayers
- *6.Drawdecorations(scrollbarsforinstance)
- */
- //Step1,drawthebackground,ifneeded
- intsaveCount;
- if(!dirtyOpaque){
- finalDrawablebackground=mBGDrawable;
- if(background!=null){
- finalintscrollX=mScrollX;
- finalintscrollY=mScrollY;
- if(mBackgroundSizeChanged){
- background.setBounds(0,0,mRight-mLeft,mBottom-mTop);
- mBackgroundSizeChanged=false;
- }
- if((scrollX|scrollY)==0){
- background.draw(canvas);
- }else{
- canvas.translate(scrollX,scrollY);
- background.draw(canvas);
- canvas.translate(-scrollX,-scrollY);
- }
- }
- }
- //skipstep2&5ifpossible(commoncase)
- finalintviewFlags=mViewFlags;
- booleanhorizontalEdges=(viewFlags&FADING_EDGE_HORIZONTAL)!=0;
- booleanverticalEdges=(viewFlags&FADING_EDGE_VERTICAL)!=0;
- if(!verticalEdges&&!horizontalEdges){
- //Step3,drawthecontent
- if(!dirtyOpaque)onDraw(canvas);
- //Step4,drawthechildren
- dispatchDraw(canvas);
- //Step6,drawdecorations(scrollbars)
- onDrawScrollBars(canvas);
- //we'redone...
- return;
- }
- /*
- *Herewedothefullfledgedroutine...
- *(thisisanuncommoncasewherespeedmattersless,
- *thisiswhywerepeatsomeoftheteststhathavebeen
- *doneabove)
- */
- booleandrawTop=false;
- booleandrawBottom=false;
- booleandrawLeft=false;
- booleandrawRight=false;
- floattopFadeStrength=0.0f;
- floatbottomFadeStrength=0.0f;
- floatleftFadeStrength=0.0f;
- floatrightFadeStrength=0.0f;
- //Step2,savethecanvas'layers
- intpaddingLeft=mPaddingLeft;
- finalbooleanoffsetRequired=isPaddingOffsetRequired();
- if(offsetRequired){
- paddingLeft+=getLeftPaddingOffset();
- }
- intleft=mScrollX+paddingLeft;
- intright=left+mRight-mLeft-mPaddingRight-paddingLeft;
- inttop=mScrollY+getFadeTop(offsetRequired);
- intbottom=top+getFadeHeight(offsetRequired);
- if(offsetRequired){
- right+=getRightPaddingOffset();
- bottom+=getBottomPaddingOffset();
- }
- finalScrollabilityCachescrollabilityCache=mScrollCache;
- finalfloatfadeHeight=scrollabilityCache.fadingEdgeLength;
- intlength=(int)fadeHeight;
- //clipthefadelengthiftopandbottomfadesoverlap
- //overlappingfadesproduceodd-lookingartifacts
- if(verticalEdges&&(top+length>bottom-length)){
- length=(bottom-top)/2;
- }
- //alsocliphorizontalfadesifnecessary
- if(horizontalEdges&&(left+length>right-length)){
- length=(right-left)/2;
- }
- if(verticalEdges){
- topFadeStrength=Math.max(0.0f,Math.min(1.0f,getTopFadingEdgeStrength()));
- drawTop=topFadeStrength*fadeHeight>1.0f;
- bottomFadeStrength=Math.max(0.0f,Math.min(1.0f,getBottomFadingEdgeStrength()));
- drawBottom=bottomFadeStrength*fadeHeight>1.0f;
- }
- if(horizontalEdges){
- leftFadeStrength=Math.max(0.0f,Math.min(1.0f,getLeftFadingEdgeStrength()));
- drawLeft=leftFadeStrength*fadeHeight>1.0f;
- rightFadeStrength=Math.max(0.0f,Math.min(1.0f,getRightFadingEdgeStrength()));
- drawRight=rightFadeStrength*fadeHeight>1.0f;
- }
- saveCount=canvas.getSaveCount();
- intsolidColor=getSolidColor();
- if(solidColor==0){
- finalintflags=Canvas.HAS_ALPHA_LAYER_SAVE_FLAG;
- if(drawTop){
- canvas.saveLayer(left,top,right,top+length,null,flags);
- }
- if(drawBottom){
- canvas.saveLayer(left,bottom-length,right,bottom,null,flags);
- }
- if(drawLeft){
- canvas.saveLayer(left,top,left+length,bottom,null,flags);
- }
- if(drawRight){
- canvas.saveLayer(right-length,top,right,bottom,null,flags);
- }
- }else{
- scrollabilityCache.setFadeColor(solidColor);
- }
- //Step3,drawthecontent
- if(!dirtyOpaque)onDraw(canvas);
- //Step4,drawthechildren
- dispatchDraw(canvas);
- //Step5,drawthefadeeffectandrestorelayers
- finalPaintp=scrollabilityCache.paint;
- finalMatrixmatrix=scrollabilityCache.matrix;
- finalShaderfade=scrollabilityCache.shader;
- if(drawTop){
- matrix.setScale(1,fadeHeight*topFadeStrength);
- matrix.postTranslate(left,top);
- fade.setLocalMatrix(matrix);
- canvas.drawRect(left,top,right,top+length,p);
- }
- if(drawBottom){
- matrix.setScale(1,fadeHeight*bottomFadeStrength);
- matrix.postRotate(180);
- matrix.postTranslate(left,bottom);
- fade.setLocalMatrix(matrix);
- canvas.drawRect(left,bottom-length,right,bottom,p);
- }
- if(drawLeft){
- matrix.setScale(1,fadeHeight*leftFadeStrength);
- matrix.postRotate(-90);
- matrix.postTranslate(left,top);
- fade.setLocalMatrix(matrix);
- canvas.drawRect(left,top,left+length,bottom,p);
- }
- if(drawRight){
- matrix.setScale(1,fadeHeight*rightFadeStrength);
- matrix.postRotate(90);
- matrix.postTranslate(right,top);
- fade.setLocalMatrix(matrix);
- canvas.drawRect(right-length,top,right,bottom,p);
- }
- canvas.restoreToCount(saveCount);
- //Step6,drawdecorations(scrollbars)
- onDrawScrollBars(canvas);
- }
同样地,将上面draw()调用流程用伪代码描述如下:
[java] view plain copy
- //ViewRoot.java
- //draw()过程
- //发起draw()的“发号者”是在ViewRoot.java里的performTraversals()方法,该方法会继续调用draw()方法开始绘图
- privatevoiddraw(){
- //...
- ViewmView;
- mView.draw(canvas);
- //....
- }
- //回调View视图里的onDraw()过程,该方法只由ViewGroup类型实现
- privatevoiddraw(Canvascanvas){
- //该方法会做如下事情
- //1、绘制该View的背景
- //2、为绘制渐变框做一些准备操作
- //3、调用onDraw()方法绘制视图本身
- //4、调用dispatchDraw()方法绘制每个子视图,dispatchDraw()已经在Android框架中实现了,在ViewGroup方法中
- //应用程序程序一般不需要重写该方法,但可以捕获该方法的发生,做一些特别的事情
- //5、绘制渐变框
- }
- //ViewGroup.java中的dispatchDraw()方法,应用程序一般不需要重写该方法
- @Override
- protectedvoiddispatchDraw(Canvascanvas){
- //
- //其实现方法类似如下:
- intchildCount=getChildCount();
- for(inti=0;i<childCount;i++){
- Viewchild=getChildAt(i);
- //调用drawChild完成
- drawChild(child,canvas);
- }
- }
- //ViewGroup.java中的dispatchDraw()方法,应用程序一般不需要重写该方法
- protectedvoiddrawChild(Viewchild,Canvascanvas){
- //....
- //简单的回调View对象的draw()方法,递归就这么产生了
- child.draw(canvas);
- //.........
- }
需要强调一点的就是,在这三个流程中,Google已经帮我们把draw()过程框架已经写好了,自定义的ViewGroup只需要实现measure()过程和layout()过程即可。
引起View树重新绘制的因素有如下几种:
1、导致视图大小发生变化;
2、导致ViewGroup重新为子视图分配位置
3、视图显示情况发生变化需要重绘
这三种情况,最终会直接或间接调用到三个方法,分别为invalidate(),requsetLaytout()以及requestFocus() ,接着这三个方法最终会调用到ViewRoot中的schedulTraversale()方法,该函数然后发起一个异步消息,消息处理中调用performTraverser()方法对整个View进行遍历。
invalidate()方法:
说明:请求重绘View树,即draw()过程,假如视图大小没有发生变化就不会调用layout()过程,并且只绘制那些“需要重绘的”视图,即谁(View的话,只绘制该View;ViewGroup,则绘制整个ViewGroup)请求invalidate()方法,就绘制该视图。
一般引起invalidate()操作的函数如下:
1、调用invalidate()方法,请求重新draw(),但只会绘制调用者本身。
2、调用setSelection()方法,请求重新draw(),但只会绘制调用者本身。
3、调用setVisibility()方法:当View可视状态在INVISIBLE转换VISIBLE时,会间接调用invalidate()方法,继而绘制该View。
4、调用setEnabled()方法:请求重新draw(),但不会重新绘制任何视图包括该调用者本身。
requestLayout()方法:
会导致调用measure()过程和layout()过程。
说明:只是对View树重新布局layout过程包括measure()和layout()过程,不会调用draw()过程,但不会重新绘制任何视图包括该调用者本身。
一般引起invalidate()操作的函数如下:
1、requestLayout()方法:当View的可视状态在INVISIBLE / VISIBLE 转换为GONE状态时,会间接调用requestLayout() 和invalidate方法。同时,由于整个个View树大小发生了变化,会请求measure()过程以及draw()过程,同样地,只绘制需要“重新绘制”的视图。
requestFocus()方法:
说明:请求View树的draw()过程,但只绘制“需要重绘”的视图。
参考资料:
1、Android中View绘制流程以及invalidate()等相关方法分析http://blog.csdn.net/qinjuning/article/details/7110211
更多相关文章
- 抽离Android原生控件的方法
- adb通过wifi连接android设备的方法
- 在 android 上运行 python 的方法
- 界面编程之基本界面组件(7)ImageView(图像视图)
- 饭后Android 第一餐-NavigationView+Toolbar(NavigationView使用
- Android jni 常用方法备忘
- Android SQLite使用方法