《Android開發藝術探索》筆記:
在Activity的onCreate()或者onResume()中去獲得View的高度的時候不能正確獲得寬度和高度資訊,這是因為 View的measure過程和Activity的生命週期不是同步執行的,因此無法保證Activity執行了onCreate onStart onResume時,某個View已經測量完畢了,如果還沒有測量完,那麼獲得的寬高就是0。可以通過下面幾種方式來獲得:
#####1、onWindowFocusChanged
onWindowFocusChanged:View已經初始化完畢,寬高已經有了,需要注意onWindowFocusChanged會被呼叫多次,Activity得到焦點和失去焦點都會執行這個回撥,見下圖:
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是合理的。