前言
可能很多情況下,我們都會有在activity中獲取view 的尺寸大小(寬度和高度)的需求。面對這種情況,很多同學立馬反應:這么簡單的問題,還用你說?你是不是傻。。然后立馬寫下getWidth()、getHeight()等方法,洋洋得意的就走了。然而事實就是這樣的嗎?實踐證明,我們這樣是獲取不到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 監(jiān)聽事件
在視圖將要繪制時調用該監(jiān)聽事件,會被調用多次,因此獲取到視圖的寬度和高度后要移除該監(jiān)聽事件。
view.getViewTreeObserver().addOnPreDrawListener( new ViewTreeObserver.OnPreDrawListener() { @Override public boolean onPreDraw() { view.getViewTreeObserver().removeOnPreDrawListener(this); view.getWidth(); // 獲取寬度 view.getHeight(); // 獲取高度 return true; }});
三、使用 ViewTreeObserver. OnGlobalLayoutListener 監(jiān)聽事件
在布局發(fā)生改變或者某個視圖的可視狀態(tài)發(fā)生改變時調用該事件,會被多次調用,因此需要在獲取到視圖的寬度和高度后執(zhí)行 remove 方法移除該監(jiān)聽事件。
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 方法
在視圖的大小發(fā)生改變時調用該方法,會被多次調用,因此獲取到寬度和高度后需要考慮禁用掉代碼。
該實現方法需要繼承 View,且多次被調用,不建議使用。
@Overrideprotected void onSizeChanged(int w, int h, int oldw, int oldh) { super.onSizeChanged(w, h, oldw, oldh); view.getWidth(); // 獲取寬度 view.getHeight(); // 獲取高度}
五、重寫 View 的 onLayout 方法
該方法會被多次調用,獲取到寬度和高度后需要考慮禁用掉代碼。
該實現方法需要繼承 View,且多次被調用,不建議使用。
@Overrideprotected 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 監(jiān)聽事件(API >= 11)
在視圖的 layout 改變時調用該事件,會被多次調用,因此需要在獲取到視圖的寬度和高度后執(zhí)行 remove 方法移除該監(jiān)聽事件。
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 等事件完成后觸發(fā)。
UI 事件隊列會按順序處理事件,在 setContentView() 被調用后,事件隊列中會包含一個要求重新 layout 的 message,所以任何 post 到隊列中的 Runnable 對象都會在 Layout 發(fā)生變化后執(zhí)行。
該方法只會執(zhí)行一次,且邏輯簡單,建議使用。
view.post(new Runnable() { @Override public void run() { view.getWidth(); // 獲取寬度 view.getHeight(); // 獲取高度 }});
以上為轉載內容,個人學習收藏記錄
下面是自己的學習記錄。
首先第一個方法,以前用過,確實不準確,猜測是應該是因為參數沒有用好,因為參數只使用UNSPECIFIED未指定的測量方式,一般像Wrap_Content,才是該測量方式。
這里貼一個比較好用的, 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代碼創(chuàng)建的view,使用上面的方法無法獲得寬高,因為我使用的是ScrollView。像在自定義中,加載一次布局,應該選中最后一個post的方法最為使用。
另外還用的多的,應該是第三種方式,一般在外部使用,比如需要等待Recyclerview繪制完成后進行的操作。
總結
以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對VEVB武林網的支持。
新聞熱點
疑難解答