Android 獲得View寬高的幾種方式

_小馬快跑_發表於2017-12-15

《Android開發藝術探索》筆記:

在Activity的onCreate()或者onResume()中去獲得View的高度的時候不能正確獲得寬度和高度資訊,這是因為 View的measure過程和Activity的生命週期不是同步執行的,因此無法保證Activity執行了onCreate onStart onResume時,某個View已經測量完畢了,如果還沒有測量完,那麼獲得的寬高就是0。可以通過下面幾種方式來獲得:

#####1、onWindowFocusChanged

onWindowFocusChanged:View已經初始化完畢,寬高已經有了,需要注意onWindowFocusChanged會被呼叫多次,Activity得到焦點和失去焦點都會執行這個回撥,見下圖:

window_focus1.png

1、Activity首次進入的時候執行的方法 2、跳轉到另一個Activity時 3、返回到當前Activity時 可見當執行onResume和onPause時,onWindowFocusChanged都會被呼叫。

 @Override
 public void onWindowFocusChanged(boolean hasFocus) {
     super.onWindowFocusChanged(hasFocus);
     if (hasFocus) {
         //獲得寬度
         int width = view.getMeasuredWidth();
         //獲得高度
         int height = view.getMeasuredHeight();
     }
 }
複製程式碼

#####2、view.post(runnable) 通過post可以將一個runnable投遞到訊息佇列的尾部,等待Looper呼叫此runnable的時候,View也已經初始化好了,示例:

 @Override
 protected void onStart() {
     super.onStart();
     view.post(new Runnable() {
         @Override
         public void run() {
             int width=view.getMeasuredWidth();
             int height=view.getMeasuredHeight();
         }
     })
 }
複製程式碼

#####3、ViewTreeObserver 使用ViewTreeObserver的眾多回撥可以完成這個功能,比如使用OnGlobalLayoutListener這個介面,當View樹的狀態發生改變或者View樹內部的View的可見性發生改變時,OnGlobalLayout方法將會被回撥,這是獲取View寬高很好的一個時機,需要注意的是,伴隨著View樹的狀態改變,OnGlobalLayout會被呼叫多次,示例:

@Override
protected void onStart() {
    super.onStart();
    ViewTreeObserver observer=view.getViewTreeObserver();
    observer.addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
        @Override
        public void onGlobalLayout() {
            view.getViewTreeObserver().removeOnGlobalLayoutListener(this);
            int width=view.getMeasuredWidth();
            int height=view.getMeasuredHeight();
        }
    });
}
複製程式碼

更詳細參看:Android ViewTreeObserver使用總結及獲得View高度的幾種方法

4、view.measure(int widthMeasureSpec, int heightMeasureSpec)

通過手動對View進行measure來得到View的寬高,這裡要分情況處理,根據View的LayoutParams來分:

  • match-parent 無法測出具體的寬高,因為根據View的measure過程,構造此種MeasureSpec需要知道parentSize,即父容器的剩餘空間,而這個值我們是不知道的,所以不能測出View的大小。

  • 具體的數值(dp/px) 比如寬高都是100px,如下measure:

 int widthMeasureSpec = View.MeasureSpec.makeMeasureSpec(100, View.MeasureSpec.EXACTLY);
 int heightMeasureSpec = View.MeasureSpec.makeMeasureSpec(100, View.MeasureSpec.EXACTLY);
 view.measure(widthMeasureSpec, heightMeasureSpec);
複製程式碼
  • wrap_content

如下measure:

 int widthMeasureSpec = View.MeasureSpec.makeMeasureSpec((1 << 30) - 1, View.MeasureSpec.AT_MOST);
 int heightMeasureSpec = View.MeasureSpec.makeMeasureSpec((1 << 30) - 1, View.MeasureSpec.AT_MOST);
 view.measure(widthMeasureSpec, heightMeasureSpec);
複製程式碼

View的specSize使用30位二進位制表示,也就是說最大是30個1,也就是(1 << 30) - 1,在最大化模式下,我們用View理論上能支援的最大值去構造MeasureSpec是合理的。

相關文章