android如何獲取view在布局中的高度與寬度詳解
前言
可能很多情況下,我們都會(huì)有在activity中獲取view 的尺寸大?。▽挾群透叨龋┑男枨蟆C鎸?duì)這種情況,很多同學(xué)立馬反應(yīng):這么簡(jiǎn)單的問(wèn)題,還用你說(shuō)?你是不是傻。。然后立馬寫(xiě)下getWidth()、getHeight()等方法,洋洋得意的就走了。然而事實(shí)就是這樣的嗎?實(shí)踐證明,我們這樣是獲取不到View的寬度和高度大小的。
當(dāng)我們?cè)?onCreate() 方法中獲取某個(gè) View 組件的寬度和高度,直接調(diào)用 getWidth()、getHeight()、getMeasuredWidth()、getMeasuredHeight() 方法只會(huì)得到 0。這是什么原因呢?下面來(lái)一起看看吧
實(shí)現(xiàn)方法
一、使用 View.measure 測(cè)量 View
該方法測(cè)量的寬度和高度可能與視圖繪制完成后的真實(shí)的寬度和高度不一致。
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)聽(tīng)事件
在視圖將要繪制時(shí)調(diào)用該監(jiān)聽(tīng)事件,會(huì)被調(diào)用多次,因此獲取到視圖的寬度和高度后要移除該監(jiān)聽(tīng)事件。
view.getViewTreeObserver().addOnPreDrawListener( new ViewTreeObserver.OnPreDrawListener() { @Override public boolean onPreDraw() { view.getViewTreeObserver().removeOnPreDrawListener(this); view.getWidth(); // 獲取寬度 view.getHeight(); // 獲取高度 return true; } });
三、使用 ViewTreeObserver. OnGlobalLayoutListener 監(jiān)聽(tīng)事件
在布局發(fā)生改變或者某個(gè)視圖的可視狀態(tài)發(fā)生改變時(shí)調(diào)用該事件,會(huì)被多次調(diào)用,因此需要在獲取到視圖的寬度和高度后執(zhí)行 remove 方法移除該監(jiān)聽(tīng)事件。
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(); // 獲取高度 } });
四、重寫(xiě) View 的 onSizeChanged 方法
在視圖的大小發(fā)生改變時(shí)調(diào)用該方法,會(huì)被多次調(diào)用,因此獲取到寬度和高度后需要考慮禁用掉代碼。
該實(shí)現(xiàn)方法需要繼承 View,且多次被調(diào)用,不建議使用。
@Override protected void onSizeChanged(int w, int h, int oldw, int oldh) { super.onSizeChanged(w, h, oldw, oldh); view.getWidth(); // 獲取寬度 view.getHeight(); // 獲取高度 }
五、重寫(xiě) View 的 onLayout 方法
該方法會(huì)被多次調(diào)用,獲取到寬度和高度后需要考慮禁用掉代碼。
該實(shí)現(xiàn)方法需要繼承 View,且多次被調(diào)用,不建議使用。
@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 監(jiān)聽(tīng)事件(API >= 11)
在視圖的 layout 改變時(shí)調(diào)用該事件,會(huì)被多次調(diào)用,因此需要在獲取到視圖的寬度和高度后執(zhí)行 remove 方法移除該監(jiān)聽(tīng)事件。
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 對(duì)象中的方法會(huì)在 View 的 measure、layout 等事件完成后觸發(fā)。
UI 事件隊(duì)列會(huì)按順序處理事件,在 setContentView() 被調(diào)用后,事件隊(duì)列中會(huì)包含一個(gè)要求重新 layout 的 message,所以任何 post 到隊(duì)列中的 Runnable 對(duì)象都會(huì)在 Layout 發(fā)生變化后執(zhí)行。
該方法只會(huì)執(zhí)行一次,且邏輯簡(jiǎn)單,建議使用。
view.post(new Runnable() { @Override public void run() { view.getWidth(); // 獲取寬度 view.getHeight(); // 獲取高度 } });
以上為轉(zhuǎn)載內(nèi)容,個(gè)人學(xué)習(xí)收藏記錄
下面是自己的學(xué)習(xí)記錄。
首先第一個(gè)方法,以前用過(guò),確實(shí)不準(zhǔn)確,猜測(cè)是應(yīng)該是因?yàn)閰?shù)沒(méi)有用好,因?yàn)閰?shù)只使用UNSPECIFIED未指定的測(cè)量方式,一般像Wrap_Content,才是該測(cè)量方式。
這里貼一個(gè)比較好用的, 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的時(shí)候,需要在一次add代碼創(chuàng)建的view,使用上面的方法無(wú)法獲得寬高,因?yàn)槲沂褂玫氖荢crollView。像在自定義中,加載一次布局,應(yīng)該選中最后一個(gè)post的方法最為使用。
另外還用的多的,應(yīng)該是第三種方式,一般在外部使用,比如需要等待Recyclerview繪制完成后進(jìn)行的操作。
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
- Android RecyclerView網(wǎng)格布局示例解析
- Android自定義ViewGroup實(shí)現(xiàn)流式布局
- Android列表RecyclerView排列布局
- Android進(jìn)階教程之ViewGroup自定義布局
- Android RecyclerView多類(lèi)型布局卡片解決方案
- Android自定View流式布局根據(jù)文字?jǐn)?shù)量換行
- Android 自定義View實(shí)現(xiàn)任意布局的RadioGroup效果
- Android控件CardView實(shí)現(xiàn)卡片布局
- Android網(wǎng)格布局GridView實(shí)現(xiàn)漂亮的多選效果
- Android RecyclerView實(shí)現(xiàn)多種item布局的方法
- 自己實(shí)現(xiàn)Android View布局流程
相關(guān)文章
Android截取指定View為圖片的實(shí)現(xiàn)方法
這篇文章主要為大家詳細(xì)介紹了Android截取指定View為圖片的實(shí)現(xiàn)方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-06-06Android Toolbar自定義標(biāo)題標(biāo)題居中的實(shí)例代碼
這篇文章主要介紹了Android Toolbar自定義標(biāo)題 標(biāo)題居中的實(shí)例代碼,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2018-08-08解析Android 8.1平臺(tái)SystemUI 導(dǎo)航欄加載流程
這篇文章主要介紹了Android 8.1平臺(tái)SystemUI 導(dǎo)航欄加載流程,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-11-11android調(diào)用WebService實(shí)例分析
這篇文章主要介紹了android調(diào)用WebService的方法,以實(shí)例形式較為詳細(xì)的分析了WebService的調(diào)用原理與具體使用方法,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-10-10簡(jiǎn)單實(shí)現(xiàn)Android計(jì)算器功能
這篇文章主要為大家詳細(xì)介紹了自己動(dòng)手實(shí)現(xiàn)的Android計(jì)算器功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-01-01Android獲取手機(jī)型號(hào)/系統(tǒng)版本號(hào)/App版本號(hào)等信息實(shí)例講解
本示例獲得手機(jī)型號(hào),系統(tǒng)版本,App版本號(hào)等信息,具體實(shí)現(xiàn)如下,感興趣的朋友可以參考下哈2013-06-06Android平臺(tái)生成二維碼并實(shí)現(xiàn)掃描 & 識(shí)別功能
這篇文章主要介紹了Android平臺(tái)生成二維碼并實(shí)現(xiàn)掃描 & 識(shí)別功能的相關(guān)資料,需要的朋友可以參考下2016-06-06