Android学习札记15:对Android中View绘制流程的一些理解

分类:Android 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
  1. Thisiscalledtofindouthowbigaviewshouldbe.Theparentsuppliesconstraintinformationinthewidthandheightparameters.
  2. TheactualmesurementworkofaviewisperformedinonMeasure(int,int),calledbythismethod.Therefore,onlyonMeasure(int,int)canandmustbe
  3. overridenbysubclasses.
  4. Parameters:
  5. widthMeasureSpecHorizontalspacerequirementsasimposedbytheparent
  6. heightMeasureSpecVerticalspacerequirementsasimposedbytheparent
  7. Seealso:
  8. onMeasure(int,int)
  9. publicfinalvoidmeasure(intwidthMeasureSpec,intheightMeasureSpec){
  10. if((mPrivateFlags&FORCE_LAYOUT)==FORCE_LAYOUT||
  11. widthMeasureSpec!=mOldWidthMeasureSpec||
  12. heightMeasureSpec!=mOldHeightMeasureSpec){
  13. //firstclearsthemeasureddimensionflag
  14. mPrivateFlags&=~MEASURED_DIMENSION_SET;
  15. if(ViewDebug.TRACE_HIERARCHY){
  16. ViewDebug.trace(this,ViewDebug.HierarchyTraceType.ON_MEASURE);
  17. }
  18. //measureourselves,thisshouldsetthemeasureddimensionflagback
  19. onMeasure(widthMeasureSpec,heightMeasureSpec);
  20. //flagnotset,setMeasuredDimension()wasnotinvoked,weraise
  21. //anexceptiontowarnthedeveloper
  22. if((mPrivateFlags&MEASURED_DIMENSION_SET)!=MEASURED_DIMENSION_SET){
  23. thrownewIllegalStateException("onMeasure()didnotsetthe"
  24. +"measureddimensionbycalling"
  25. +"setMeasuredDimension()");
  26. }
  27. mPrivateFlags|=LAYOUT_REQUIRED;
  28. }
  29. mOldWidthMeasureSpec=widthMeasureSpec;
  30. mOldHeightMeasureSpec=heightMeasureSpec;
  31. }


为了大家更好的理解,采用“二B程序员”的方式利用伪代码描述该measure流程


[java] view plain copy
  1. //ViewRoot.java
  2. //measure()过程
  3. //发起measure()过程的"发号者"是在ViewRoot.java里的performTraversals()方法中的mView.measure()
  4. privatevoidperformTraversals(){
  5. //...
  6. ViewmView;
  7. mView.measure(h,l);
  8. //....
  9. }
  10. //回调View视图里的onMeasure过程
  11. privatevoidonMeasure(intheight,intwidth){
  12. //设置该view的实际高(mMeasuredHeight)和宽(mMeasuredWidth)
  13. //1、该方法必须在onMeasure调用,否者报异常。
  14. setMeasuredDimension(h,l);
  15. //2、如果该View是ViewGroup类型,则对它的每个子View进行measure()过程
  16. intchildCount=getChildCount();
  17. for(inti=0;i<childCount;i++){
  18. //2.1、获得每个子View对象引用
  19. Viewchild=getChildAt(i);
  20. //整个measure()过程就是个递归过程
  21. //该方法只是一个过滤器,最后会调用measure()过程或者measureChild(child,h,i)方法
  22. measureChildWithMargins(child,h,i);
  23. //其实,对于我们自己写的应用来说,最好的办法是去掉框架里的该方法,直接调用view.measure(),如下:
  24. //child.measure(h,l);
  25. }
  26. }
  27. //该方法在ViewGroup.java里具体实现
  28. protectedvoidmeasureChildWithMargins(Viewv,intheight,intwidth){
  29. v.measure(h,l);
  30. }


流程二: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
  1. Assignasizeandpositiontoaviewandallofitsdescendants
  2. Thisisthesecondphaseofthelayoutmechanism.(Thefirstismeasuring).Inthisphase,eachparentcallslayoutonallofitschildrentopositionthem.
  3. Thisistypicallydoneusingthechildmeasurementsthatwerestoredinthemeasurepass().
  4. Derivedclassesshouldnotoverridethismethod.DerivedclasseswithchildrenshouldoverrideonLayout.Inthatmethod,theyshouldcalllayoutoneach
  5. oftheirchildren.
  6. Parameters:
  7. lLeftposition,relativetoparent
  8. tTopposition,relativetoparent
  9. rRightposition,relativetoparent
  10. bBottomposition,relativetoparent
  11. @SuppressWarnings({"unchecked"})
  12. publicvoidlayout(intl,intt,intr,intb){
  13. intoldL=mLeft;
  14. intoldT=mTop;
  15. intoldB=mBottom;
  16. intoldR=mRight;
  17. booleanchanged=setFrame(l,t,r,b);//设置每个视图位于父视图的位置坐标
  18. if(changed||(mPrivateFlags&LAYOUT_REQUIRED)==LAYOUT_REQUIRED){
  19. if(ViewDebug.TRACE_HIERARCHY){
  20. ViewDebug.trace(this,ViewDebug.HierarchyTraceType.ON_LAYOUT);
  21. }
  22. onLayout(changed,l,t,r,b);//回调onLayout()方法,设置每个子视图的布局
  23. mPrivateFlags&=~LAYOUT_REQUIRED;
  24. ListenerInfoli=mListenerInfo;
  25. if(li!=null&&li.mOnLayoutChangeListeners!=null){
  26. ArrayList<OnLayoutChangeListener>listenersCopy=
  27. (ArrayList<OnLayoutChangeListener>)li.mOnLayoutChangeListeners.clone();
  28. intnumListeners=listenersCopy.size();
  29. for(inti=0;i<numListeners;++i){
  30. listenersCopy.get(i).onLayoutChange(this,l,t,r,b,oldL,oldT,oldR,oldB);
  31. }
  32. }
  33. }
  34. mPrivateFlags&=~FORCE_LAYOUT;
  35. }


同样地,将上面layout()调用流程用伪代码描述如下:


[java] view plain copy
  1. //ViewRoot.java
  2. //layout()过程
  3. //发起layout()过程的“发号者”是在ViewRoot.java里的performTraversals()方法中的mView.layout()
  4. privatevoidperformTraversals(){
  5. //...
  6. ViewmView;
  7. mView.layout(left,top,right,bottom);
  8. //....
  9. }
  10. //回调View视图里的onLayout过程,该方法只由ViewGroup类型实现
  11. privatevoidonLayout(intleft,inttop,right,bottom){
  12. //如果该View不是ViewGroup类型
  13. //调用setFrame()方法设置该控件的在父视图上的坐标轴
  14. setFrame(l,t,r,b);
  15. //--------------------------
  16. //如果该View是ViewGroup类型,则对它的每个子View进行layout()过程
  17. intchildCount=getChildCount();
  18. for(inti=0;i<childCount;i++){
  19. //获得每个子View对象引用
  20. Viewchild=getChildAt(i);
  21. //整个layout()过程就是个递归过程
  22. child.layout(l,t,r,b);
  23. }
  24. }


流程三: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
  1. /**
  2. *Manuallyrenderthisview(andallofitschildren)tothegivenCanvas.
  3. *Theviewmusthavealreadydoneafulllayoutbeforethisfunctionis
  4. *called.Whenimplementingaview,implement
  5. *{@link#onDraw(android.graphics.Canvas)}insteadofoverridingthismethod.
  6. *Ifyoudoneedtooverridethismethod,callthesuperclassversion.
  7. *
  8. *@paramcanvasTheCanvastowhichtheViewisrendered.
  9. */
  10. publicvoiddraw(Canvascanvas){
  11. if(ViewDebug.TRACE_HIERARCHY){
  12. ViewDebug.trace(this,ViewDebug.HierarchyTraceType.DRAW);
  13. }
  14. finalintprivateFlags=mPrivateFlags;
  15. finalbooleandirtyOpaque=(privateFlags&DIRTY_MASK)==DIRTY_OPAQUE&&
  16. (mAttachInfo==null||!mAttachInfo.mIgnoreDirtyState);
  17. mPrivateFlags=(privateFlags&~DIRTY_MASK)|DRAWN;
  18. /*
  19. *Drawtraversalperformsseveraldrawingstepswhichmustbeexecuted
  20. *intheappropriateorder:
  21. *
  22. *1.Drawthebackground
  23. *2.Ifnecessary,savethecanvas'layerstoprepareforfading
  24. *3.Drawview'scontent
  25. *4.Drawchildren
  26. *5.Ifnecessary,drawthefadingedgesandrestorelayers
  27. *6.Drawdecorations(scrollbarsforinstance)
  28. */
  29. //Step1,drawthebackground,ifneeded
  30. intsaveCount;
  31. if(!dirtyOpaque){
  32. finalDrawablebackground=mBGDrawable;
  33. if(background!=null){
  34. finalintscrollX=mScrollX;
  35. finalintscrollY=mScrollY;
  36. if(mBackgroundSizeChanged){
  37. background.setBounds(0,0,mRight-mLeft,mBottom-mTop);
  38. mBackgroundSizeChanged=false;
  39. }
  40. if((scrollX|scrollY)==0){
  41. background.draw(canvas);
  42. }else{
  43. canvas.translate(scrollX,scrollY);
  44. background.draw(canvas);
  45. canvas.translate(-scrollX,-scrollY);
  46. }
  47. }
  48. }
  49. //skipstep2&5ifpossible(commoncase)
  50. finalintviewFlags=mViewFlags;
  51. booleanhorizontalEdges=(viewFlags&FADING_EDGE_HORIZONTAL)!=0;
  52. booleanverticalEdges=(viewFlags&FADING_EDGE_VERTICAL)!=0;
  53. if(!verticalEdges&&!horizontalEdges){
  54. //Step3,drawthecontent
  55. if(!dirtyOpaque)onDraw(canvas);
  56. //Step4,drawthechildren
  57. dispatchDraw(canvas);
  58. //Step6,drawdecorations(scrollbars)
  59. onDrawScrollBars(canvas);
  60. //we'redone...
  61. return;
  62. }
  63. /*
  64. *Herewedothefullfledgedroutine...
  65. *(thisisanuncommoncasewherespeedmattersless,
  66. *thisiswhywerepeatsomeoftheteststhathavebeen
  67. *doneabove)
  68. */
  69. booleandrawTop=false;
  70. booleandrawBottom=false;
  71. booleandrawLeft=false;
  72. booleandrawRight=false;
  73. floattopFadeStrength=0.0f;
  74. floatbottomFadeStrength=0.0f;
  75. floatleftFadeStrength=0.0f;
  76. floatrightFadeStrength=0.0f;
  77. //Step2,savethecanvas'layers
  78. intpaddingLeft=mPaddingLeft;
  79. finalbooleanoffsetRequired=isPaddingOffsetRequired();
  80. if(offsetRequired){
  81. paddingLeft+=getLeftPaddingOffset();
  82. }
  83. intleft=mScrollX+paddingLeft;
  84. intright=left+mRight-mLeft-mPaddingRight-paddingLeft;
  85. inttop=mScrollY+getFadeTop(offsetRequired);
  86. intbottom=top+getFadeHeight(offsetRequired);
  87. if(offsetRequired){
  88. right+=getRightPaddingOffset();
  89. bottom+=getBottomPaddingOffset();
  90. }
  91. finalScrollabilityCachescrollabilityCache=mScrollCache;
  92. finalfloatfadeHeight=scrollabilityCache.fadingEdgeLength;
  93. intlength=(int)fadeHeight;
  94. //clipthefadelengthiftopandbottomfadesoverlap
  95. //overlappingfadesproduceodd-lookingartifacts
  96. if(verticalEdges&&(top+length>bottom-length)){
  97. length=(bottom-top)/2;
  98. }
  99. //alsocliphorizontalfadesifnecessary
  100. if(horizontalEdges&&(left+length>right-length)){
  101. length=(right-left)/2;
  102. }
  103. if(verticalEdges){
  104. topFadeStrength=Math.max(0.0f,Math.min(1.0f,getTopFadingEdgeStrength()));
  105. drawTop=topFadeStrength*fadeHeight>1.0f;
  106. bottomFadeStrength=Math.max(0.0f,Math.min(1.0f,getBottomFadingEdgeStrength()));
  107. drawBottom=bottomFadeStrength*fadeHeight>1.0f;
  108. }
  109. if(horizontalEdges){
  110. leftFadeStrength=Math.max(0.0f,Math.min(1.0f,getLeftFadingEdgeStrength()));
  111. drawLeft=leftFadeStrength*fadeHeight>1.0f;
  112. rightFadeStrength=Math.max(0.0f,Math.min(1.0f,getRightFadingEdgeStrength()));
  113. drawRight=rightFadeStrength*fadeHeight>1.0f;
  114. }
  115. saveCount=canvas.getSaveCount();
  116. intsolidColor=getSolidColor();
  117. if(solidColor==0){
  118. finalintflags=Canvas.HAS_ALPHA_LAYER_SAVE_FLAG;
  119. if(drawTop){
  120. canvas.saveLayer(left,top,right,top+length,null,flags);
  121. }
  122. if(drawBottom){
  123. canvas.saveLayer(left,bottom-length,right,bottom,null,flags);
  124. }
  125. if(drawLeft){
  126. canvas.saveLayer(left,top,left+length,bottom,null,flags);
  127. }
  128. if(drawRight){
  129. canvas.saveLayer(right-length,top,right,bottom,null,flags);
  130. }
  131. }else{
  132. scrollabilityCache.setFadeColor(solidColor);
  133. }
  134. //Step3,drawthecontent
  135. if(!dirtyOpaque)onDraw(canvas);
  136. //Step4,drawthechildren
  137. dispatchDraw(canvas);
  138. //Step5,drawthefadeeffectandrestorelayers
  139. finalPaintp=scrollabilityCache.paint;
  140. finalMatrixmatrix=scrollabilityCache.matrix;
  141. finalShaderfade=scrollabilityCache.shader;
  142. if(drawTop){
  143. matrix.setScale(1,fadeHeight*topFadeStrength);
  144. matrix.postTranslate(left,top);
  145. fade.setLocalMatrix(matrix);
  146. canvas.drawRect(left,top,right,top+length,p);
  147. }
  148. if(drawBottom){
  149. matrix.setScale(1,fadeHeight*bottomFadeStrength);
  150. matrix.postRotate(180);
  151. matrix.postTranslate(left,bottom);
  152. fade.setLocalMatrix(matrix);
  153. canvas.drawRect(left,bottom-length,right,bottom,p);
  154. }
  155. if(drawLeft){
  156. matrix.setScale(1,fadeHeight*leftFadeStrength);
  157. matrix.postRotate(-90);
  158. matrix.postTranslate(left,top);
  159. fade.setLocalMatrix(matrix);
  160. canvas.drawRect(left,top,left+length,bottom,p);
  161. }
  162. if(drawRight){
  163. matrix.setScale(1,fadeHeight*rightFadeStrength);
  164. matrix.postRotate(90);
  165. matrix.postTranslate(right,top);
  166. fade.setLocalMatrix(matrix);
  167. canvas.drawRect(right-length,top,right,bottom,p);
  168. }
  169. canvas.restoreToCount(saveCount);
  170. //Step6,drawdecorations(scrollbars)
  171. onDrawScrollBars(canvas);
  172. }


同样地,将上面draw()调用流程用伪代码描述如下:


[java] view plain copy
  1. //ViewRoot.java
  2. //draw()过程
  3. //发起draw()的“发号者”是在ViewRoot.java里的performTraversals()方法,该方法会继续调用draw()方法开始绘图
  4. privatevoiddraw(){
  5. //...
  6. ViewmView;
  7. mView.draw(canvas);
  8. //....
  9. }
  10. //回调View视图里的onDraw()过程,该方法只由ViewGroup类型实现
  11. privatevoiddraw(Canvascanvas){
  12. //该方法会做如下事情
  13. //1、绘制该View的背景
  14. //2、为绘制渐变框做一些准备操作
  15. //3、调用onDraw()方法绘制视图本身
  16. //4、调用dispatchDraw()方法绘制每个子视图,dispatchDraw()已经在Android框架中实现了,在ViewGroup方法中
  17. //应用程序程序一般不需要重写该方法,但可以捕获该方法的发生,做一些特别的事情
  18. //5、绘制渐变框
  19. }
  20. //ViewGroup.java中的dispatchDraw()方法,应用程序一般不需要重写该方法
  21. @Override
  22. protectedvoiddispatchDraw(Canvascanvas){
  23. //
  24. //其实现方法类似如下:
  25. intchildCount=getChildCount();
  26. for(inti=0;i<childCount;i++){
  27. Viewchild=getChildAt(i);
  28. //调用drawChild完成
  29. drawChild(child,canvas);
  30. }
  31. }
  32. //ViewGroup.java中的dispatchDraw()方法,应用程序一般不需要重写该方法
  33. protectedvoiddrawChild(Viewchild,Canvascanvas){
  34. //....
  35. //简单的回调View对象的draw()方法,递归就这么产生了
  36. child.draw(canvas);
  37. //.........
  38. }



需要强调一点的就是,在这三个流程中,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

更多相关文章

  1. 抽离Android原生控件的方法
  2. adb通过wifi连接android设备的方法
  3. 在 android 上运行 python 的方法
  4. 界面编程之基本界面组件(7)ImageView(图像视图)
  5. 饭后Android 第一餐-NavigationView+Toolbar(NavigationView使用
  6. Android jni 常用方法备忘
  7. Android SQLite使用方法

随机推荐

  1. mysql触发器原理与用法实例分析
  2. mysql索引原理与用法实例分析
  3. mysql视图原理与用法实例详解
  4. Win10安装MySQL8压缩包版的教程
  5. mysql外键基本功能与用法详解
  6. mysql连接查询、联合查询、子查询原理与
  7. Mysql数据库设计三范式实例解析
  8. mysql数据类型和字段属性原理与用法详解
  9. 简单了解MYSQL数据库优化阶段
  10. Windows下MySQL主从复制的配置方法