android中獲取view在佈局中的高度和寬度
有時需要在 onCreate() 方法中獲取某個 View 元件的寬度和高度,而直接呼叫 getWidth()、getHeight()、getMeasuredWidth()、getMeasuredHeight() 方法只會得到 0。
實現方法
一、使用 View.measure 測量 View
該方法測量的寬度和高度可能與檢視繪製完成後的真實的寬度和高度不一致。
int width = View.MeasureSpec.makeMeasureSpec(0, View.MeasureSpec.UNSPECIFIED); int height = View.MeasureSpec.makeMeasureSpec(0, View.MeasureSpec.UNSPECIFIED); view.measure(width, height); view.getMeasuredWidth(); // 獲取寬度 view.getMeasuredHeight(); // 獲取高度
二、使用 ViewTreeObserver. OnPreDrawListener 監聽事件
在檢視將要繪製時呼叫該監聽事件,會被呼叫多次,因此獲取到檢視的寬度和高度後要移除該監聽事件。
view.getViewTreeObserver().addOnPreDrawListener( new ViewTreeObserver.OnPreDrawListener() { @Override public boolean onPreDraw() { view.getViewTreeObserver().removeOnPreDrawListener(this); view.getWidth(); // 獲取寬度 view.getHeight(); // 獲取高度 return true; } });
三、使用 ViewTreeObserver. OnGlobalLayoutListener 監聽事件
在佈局發生改變或者某個檢視的可視狀態發生改變時呼叫該事件,會被多次呼叫,因此需要在獲取到檢視的寬度和高度後執行 remove 方法移除該監聽事件。
view.getViewTreeObserver().addOnGlobalLayoutListener( new ViewTreeObserver.OnGlobalLayoutListener() { @Override public void onGlobalLayout() { if (Build.VERSION.SDK_INT >= 16) { view.getViewTreeObserver() .removeOnGlobalLayoutListener(this); } else { view.getViewTreeObserver() .removeGlobalOnLayoutListener(this); } view.getWidth(); // 獲取寬度 view.getHeight(); // 獲取高度 } });
四、重寫 View 的 onSizeChanged 方法
在檢視的大小發生改變時呼叫該方法,會被多次呼叫,因此獲取到寬度和高度後需要考慮禁用掉程式碼。
該實現方法需要繼承 View,且多次被呼叫,不建議使用。
@Override protected void onSizeChanged(int w, int h, int oldw, int oldh) { super.onSizeChanged(w, h, oldw, oldh); view.getWidth(); // 獲取寬度 view.getHeight(); // 獲取高度 }
五、重寫 View 的 onLayout 方法
該方法會被多次呼叫,獲取到寬度和高度後需要考慮禁用掉程式碼。
該實現方法需要繼承 View,且多次被呼叫,不建議使用。
@Override protected void onLayout(boolean changed, int l, int t, int r, int b) { super.onLayout(changed, l, t, r, b); view.getWidth(); // 獲取寬度 view.getHeight(); // 獲取高度 }
六、使用 View.OnLayoutChangeListener 監聽事件(API >= 11)
在檢視的 layout 改變時呼叫該事件,會被多次呼叫,因此需要在獲取到檢視的寬度和高度後執行 remove 方法移除該監聽事件。
view.addOnLayoutChangeListener( new View.OnLayoutChangeListener() { @Override public void onLayoutChange(View v, int l, int t, int r, int b, int oldL, int oldT, int oldR, int oldB) { view.removeOnLayoutChangeListener(this); view.getWidth(); // 獲取寬度 view.getHeight(); // 獲取高度 } });
七、使用 View.post() 方法
Runnable 物件中的方法會在 View 的 measure、layout 等事件完成後觸發。
UI 事件佇列會按順序處理事件,在 setContentView() 被呼叫後,事件佇列中會包含一個要求重新 layout 的 message,所以任何 post 到佇列中的 Runnable 物件都會在 Layout 發生變化後執行。
該方法只會執行一次,且邏輯簡單,建議使用。
view.post(new Runnable() { @Override public void run() { view.getWidth(); // 獲取寬度 view.getHeight(); // 獲取高度 } });
以上為轉載內容,個人學習收藏記錄,ofollow,noindex">原文地址
下面是自己的學習記錄。
首先第一個方法,以前用過,確實不準確,猜測是應該是因為引數沒有用好,因為引數只使用UNSPECIFIED未指定的測量方式,一般像Wrap_Content,才是該測量方式。
這裡貼一個比較好用的,AndroidUtilCode" target="_blank" rel="nofollow,noindex">AndroidUtilCode 收藏的方法。
public static int[] measureView(final View view) { ViewGroup.LayoutParams lp = view.getLayoutParams(); if (lp == null) { lp = new ViewGroup.LayoutParams( ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT ); } int widthSpec = ViewGroup.getChildMeasureSpec(0, 0, lp.width); int lpHeight = lp.height; int heightSpec; if (lpHeight > 0) { heightSpec = View.MeasureSpec.makeMeasureSpec(lpHeight, View.MeasureSpec.EXACTLY); } else { heightSpec = View.MeasureSpec.makeMeasureSpec(0, View.MeasureSpec.UNSPECIFIED); } view.measure(widthSpec, heightSpec); return new int[]{view.getMeasuredWidth(), view.getMeasuredHeight()}; }
然後是自己在做自定義view的時候,需要在一次add程式碼建立的view,使用上面的方法無法獲得寬高,因為我使用的是ScrollView。像在自定義中,載入一次佈局,應該選中最後一個post的方法最為使用。
另外還用的多的,應該是第三種方式,一般在外部使用,比如需要等待Recyclerview繪製完成後進行的操作。