Android - View繪圖原理總結

來源:互聯網
上載者:User

標籤:告訴   eth   rac   封裝   patch   img   .com   bst   net   

Android系統的視圖結構的設計也採用了組合模式,即View作為所有圖形的基類,Viewgroup對View繼承擴充為視圖容器類,由此就得到了視圖部分的基本結構--樹形結構

View定義了繪圖的基本操作

基本操作由三個函數完成:measure()、layout()、draw(),其內部又分別包含了onMeasure()、onLayout()、onDraw()三個子方法。具體操作如下:

1、measure操作

     measure操作主要用於計算視圖的大小,即視圖的寬度和長度。在view中定義為final類型,要求子類不能修改。measure()函數中又會調用下面的函數:

     (1)onMeasure(),視圖大小的將在這裡最終確定,也就是說measure只是對onMeasure的一個封裝,子類可以覆寫 onMeasure()方法實現自己的計算視圖大小的方式,並通過setMeasuredDimension(width, height)儲存計算結果。

2、layout操作

     layout操作用於設定視圖在螢幕中顯示的位置。在view中定義為final類型,要求子類不能修改。layout()函數中有兩個基本操作:

     (1)setFrame(l,t,r,b),l,t,r,b即子視圖在父視圖中的具體位置,該函數用於將這些參數儲存起來;

     (2)onLayout(),在View中這個函數什麼都不會做,提供該函數主要是為viewGroup類型布局子視圖用的;

3、draw操作

     draw操作利用前兩部得到的參數,將視圖顯示在螢幕上,到這裡也就完成了整個的視圖繪製工作。子類也不應該修改該方法,因為其內部定義了繪圖的基本操作:

     (1)繪製背景;

     (2)如果要視圖顯示漸層框,這裡會做一些準備工作;

     (3)繪製視圖本身,即調用onDraw()函數。在view中onDraw()是個空函數,也就是說具體的視圖都要覆寫該函數來實現自己的顯示(比如 TextView在這裡實現了繪製文字的過程)。而對於ViewGroup則不需要實現該函數,因為作為容器是“沒有內容“的,其包含了多個子view, 而子View已經實現了自己的繪製方法,因此只需要告訴子view繪製自己就可以了,也就是下面的dispatchDraw()方法;

     (4)繪製子視圖,即dispatchDraw()函數。在view中這是個空函數,具體的視圖不需要實現該方法,它是專門為容器類準備的,也就是容器類必須實現該方法;

     (5)如果需要(應用程式調用了setVerticalFadingEdge或者setHorizontalFadingEdge),開始繪製漸層框;

     (6)繪製捲軸;

      從上面可以看出自訂View需要最少覆寫onMeasure()和onDraw()兩個方法。

ViewGroup中的擴充操作:

     首先Viewgroup是一個抽象類別。

1、對子視圖的measure過程

     (1)measureChildren(),內部使用一個for迴圈對子視圖進行遍曆,分別調用子視圖的measure()方法;

     (2)measureChild(),為指定的子視圖measure,會被 measureChildren調用;

     (3)measureChildWithMargins(),為指定子視圖考慮了margin和padding的measure;

      以上三個方法是ViewGroup提供的3個對子view進行測量的參考方法,設計者需要在實際中首先覆寫onMeasure(),之後再對子view進行遍曆measure,這時候就可以使用以上三個方法,當然也可以自訂方法進行遍曆。

2、對子視圖的layout過程

     在ViewGroup中onLayout()被定義為abstract類型,也就是具體的容器必須實現此方法來安排子視圖的布局位置,實現中主要考慮的是視圖的大小及視圖間的相對位置關係,如gravity、layout_gravity。

3、對子視圖的draw過程

   (1)dispatchDraw(), 該方法用於對子視圖進行遍曆然後分別讓子視圖分別draw,方法內部會首先處理布局動畫(也就是說布局動畫是在這裡處理的),如果有布局動畫則會為每個子 視圖產生一個繪製時間,之後再有一個for迴圈對子視圖進行遍曆,來調用子視圖的draw方法(實際為下邊的drawChild());

    (2)drawChild(),該方法用於具體調用子視圖的draw方法,內部首先會處理視圖動畫(也就是說視圖動畫是在這裡處理的),之後調用子視圖的draw()。

    從上面分析可以看出自訂viewGroup的時候需要最少覆寫onMeasure()和onLayout()方法,其中onMeasure方法中可以直 接調用measureChildren等已有的方法,而onLayout方法就需要設計者進行完整的定義;一般不需要覆寫以dispatchDraw() 和drawChild()這兩個方法,因為上面兩個方法已經完成了基本的事情。但是可以通過覆寫在該基礎之上做一些特殊的效果,比如

 

@Overrideprotected void dispatchDraw(Canvas canvas) {// TODO Auto-generated method stub////可以在這裡先做一些處理,包括對傳入的canvas//super.dispatchDraw(canvas);//這裡會調用drawChild繪製子視圖////所有子視圖都繪製完成後這裡還可以做一些處理,比如繪製陰影什麼的//}

 

其他

      從以上分析可以看出View樹的繪製是一個遞迴的過程,從ViewGroup一直向下遍曆,直到所有的子view都完成繪製,那這一切的源頭在什麼地方 (是誰最發起measure、layout和draw的)?當然就是在View樹的源頭了——ViewRoot!,ViewRoot中包含了視窗的總容器 DecorView,ViewRoot中的performTraversal()方法會依次調用decorView的measure、layout、 draw方法,從而完成view樹的繪製。

     invalidate()方法

     invalidate()方法會導致View樹的重新繪製,而且view中的狀態標誌mPrivateFlags中有一個關於當前視圖是否需要重繪的標 志位DRAWN,也就是說只有標誌位DRAWN置位的視圖才需要進行重繪。當視圖調用invalidate()方法時,首先會將當前視圖的DRAWN標誌 置位,之後有一個迴圈調用parent.invalidateChildinParent(),這樣會導致從當前視圖依次向上遍曆直到根視圖 ViewRoot,這個過程會將需要重繪的視表徵圖記DRAWN置位,之後ViewRoot調用performTraversals()方法,完成視圖的繪 制過程。

 

參考書籍《Android核心剖析》——柯元旦

 

https://www.cnblogs.com/xgjblog/p/4128751.html

Android - View繪圖原理總結

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.