編輯:關於Android編程
接著上次的問題,已經介紹過,在初始化或者說OnCreate方法中獲取加載的布局的寬高,最後說到,調用view.measure(0, 0);然後在調用getMeasuredWidth和getMeasuredHeight就可以獲得測量的寬高。可以參考:Android如何在初始化的時候獲取加載的布局的寬高
今天在寫類似的效果時,給ListView加載一個頭部視圖,通過listView$addHeadView添加到ListView的頭部。為了描述起來簡單起見,這個頭部視圖的布局我做了簡化為下面:headview.xml
然後在初始化的時候想獲取該布局的寬高,就這樣做了:
//布局加載器 private LayoutInflater inflater; //頭部的View private View headView; //頭部View的寬 private int headViewWidth; //頭部View的高 private int headViewHeight; inflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE); headView = inflater.inflate(R.layout.headview, null); headView.measure(0, 0); headViewWidth = headView.getMeasuredWidth(); headViewHeight = headView.getMeasuredHeight(); Log.i(TAG, "headViewWidth-->" + headViewWidth + " headViewHeight-->" + headViewHeight); addHeaderView(headView);
結果。。。。。
程序崩潰了!尼瑪,不是說好的headView.measure(0, 0);調用完就能獲取到布局的寬高了嘛,怎麼這句好報錯空指針異常了?
然後把布局修改成線性布局<喎?/kf/ware/vc/" target="_blank" class="keylink">vcD4KPHA+PHByZSBjbGFzcz0="brush:java;">
竟然就好了。。。
也獲取到了測量的寬高:
接下來又開始了對問題的研究。為了簡化問題,還是按照上次的此路。在上次的基礎上添加布局viewgroup_relativelayout.xml,其實就是將外層的LinearLayout修改成RelativeLayout了,其他地方未變。加載該布局,然後測量寬高。
看結果:
View,textview和線性布局的ViewGroup和上次的結果一樣。但是將相對布局的ViewGroup直接掛掉。。剛開始還是特別的不爽的,為什麼經常遇到問題,不過一想,解決問題的過程就是水平提高的過程,所以打起精神進行研究。嘿嘿~
通過比較發現RelativeLayout出錯了,所以很容易將空指針異常的地方定位在RelativeLayout的onMeasure函數中。
但是這個函數比較復雜了。兩百多行代碼,裡面牽扯了很多的域變量和函數,讀起來挺累的。因此在網上搜了好久,在eoe上面看到有人說了下面的方法:
如果使用Inflater的情況下會出現以上錯誤,原因是用Inflater渲染組件的時候並沒有給其指定父控件,所以渲染器不會去解析width 和 height屬性,就會導致空指針異常。
解決方法:
view.setLayoutParams(newLayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT));指定寬度和高度,然後調用measure方法就OK。
其實這個說法前面就研究過,確實是這樣子的,但是LinearLayout為什麼沒出問題RelativeLayout就出問題了呢,所以還是沒有說明白原因在哪裡。
我嘗試了一下確實搞定了,原因尚不清楚,那就肯定需要查源代碼了。在這個指導下給我提供了方向就是很可能在onMeasure方法裡面調用了該相對布局的參數mLayoutParams,由於本來為null,那麼調用的話就會報告空指針異常!
接下來就在源碼的onMeasure函數裡面將所有出現mLayoutParams的地方打上斷點。然後斷點調試
RelativeLayou類(我查看的是4.2的源碼)
//493行 if (mLayoutParams.width >= 0) { width = Math.max(width, mLayoutParams.width); } //523行 if (mLayoutParams.height >= 0) { height = Math.max(height, mLayoutParams.height); }
果然mLayoutParams變量不存在為null。
到現在為止問題就清楚了。由於調用headView.measure(0, 0);的時候是通過inflate(R.layout.headview, null);方式加載的布局,因此設置的外層RelativeLayout布局的LayoutParams是null的,恰巧相對布局沒有檢查是否為null就直接調用了mLayoutParams.width或者mLayoutParams.height,因此就報空指針錯誤了。這可能是一個bug吧我認為。
既然知道了問題是出在了這裡了,那麼也很容易解決了。
至少有以下的解決方法:
1,加載布局的時候通過inflater.inflate(R.layout.viewgroup_relativelayout,(ViewGroup) findViewById(R.id.mainLayout),false);方式加載。這樣的話會設置他的布局參數。
2 手動設置布局參數添加上
lp = new ViewGroup.LayoutParams(ViewGroup.LayoutParams.FILL_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT); setLayoutParams(lp);
現在再來看上一篇博客裡面講的這個方法:
private void measureView(View child) { ViewGroup.LayoutParams lp = child.getLayoutParams(); if(lp == null){ lp = new ViewGroup.LayoutParams(ViewGroup.LayoutParams.FILL_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT); // child.setLayoutParams(lp); } //headerView的寬度信息 int childMeasureWidth = ViewGroup.getChildMeasureSpec(0, 0, lp.width); int childMeasureHeight; if(lp.height > 0){ childMeasureHeight = MeasureSpec.makeMeasureSpec(lp.height, MeasureSpec.EXACTLY); } else { childMeasureHeight = MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED);//未指定 } //將寬和高設置給child child.measure(childMeasureWidth, childMeasureHeight); }
看起來貌似就是為了避免外層布局是RelativeLayout的情況對LayoutParams做了判斷!但是直接調用上面這個方法仍然不行,因為new出來的LayoutParams沒有設置進去,所以還需要調用child.setLayoutParams(lp);然後就搞定了。經過測試ok!
回到博客開始在ListView添加頭View所遇到的問題也就很容易解決了。
1,將相對布局換成線性布局。
2,仍然使用相對布局,使用上面說的兩種方法設置外層的相對布局的布局參數。比如使用下面的方法,在家在布局的時候帶上父布局,結果搞定!好開心
inflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE); headView = inflater.inflate(R.layout.headview, this,false); headView.measure(0, 0); //換成方法measureView (headView)也可以 headViewWidth = headView.getMeasuredWidth(); headViewHeight = headView.getMeasuredHeight(); Log.i(TAG, "headViewWidth-->" + headViewWidth + " headViewHeight-->" + headViewHeight); addHeaderView(headView);
將headView.measure(0, 0);換成measureView (headView);也可以的。
但是加載布局的時候不帶上父布局,調用的是headView =inflater.inflate(R.layout.headview, null);即使用下面代碼:
inflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE); headView = inflater.inflate(R.layout.headview, null); //headView.measure(0, 0); measureView(headView); headViewWidth = headView.getMeasuredWidth(); headViewHeight = headView.getMeasuredHeight(); Log.i(TAG, "headViewWidth-->" + headViewWidth + " headViewHeight-->" + headViewHeight); addHeaderView(headView);
程序竟然又崩潰了。看調試結果:
不過不用擔心,這時的錯誤不是空指針了,而是類轉換異常,相信通過錯誤很容易修改了
lp= new AbsListView.LayoutParams(AbsListView.LayoutParams.FILL_PARENT,AbsListView.LayoutParams.WRAP_CONTENT);
然後在運行就好了,大功告成。最後的忠告就是需要自己需要測量布局寬高的時候慎用RelativeLayout,如同解決ScrollView和ListView沖突的時候一種解決方法,這裡我們只討論其中一個沖突就是:導致ListView的高度顯示不完整,出現只顯示部分內容的情況。解決方法就是就是測量ListView的總高度,然後設置一下,關鍵代碼如下。試想,如果這時候如果各個item是RelativeLayout的話,肯定會出錯。
private void setListViewHeight(ListView listView) { ListAdapter listAdapter = listView.getAdapter(); if (listAdapter == null) { return; } int totalHeight = 0; for (int i = 0; i < listAdapter.getCount(); i++) { View listItem = listAdapter.getView(i, null, listView); listItem.measure(0, 0); totalHeight += listItem.getMeasuredHeight(); } ViewGroup.LayoutParams params = listView.getLayoutParams(); params.height = totalHeight + (listView.getDividerHeight() * (listAdapter.getCount() - 1)); listView.setLayoutParams(params); }
最後,這些東西可能有些人寫程序的時候都從沒遇到過,當然也不影響寫出漂亮的APP。但是我認為不管是什麼問題,都能鍛煉解決問題的能力,都是有助於學習。
文字表達能力太差,表述能力是硬傷啊!囧…剛開始寫博客,感謝大家圍觀。
測試加載各種類型布局的寬高參數的Demo和ListView添加頭視圖為相對布局的Demo
點此下載代碼
Android作為目前主流的移動操作系統,完全符合SQLite占用資源少的優勢,故在Android平台上,集成了一個嵌入式關系型數據庫—SQLite。如果想要開發 And
之前寫過一篇文章Android TextView 橫豎排切換(字方向不變) 是自定義了一個LinearLayout, 實現了當然還不夠, 還要對它進行操作, 平移,旋轉
之前開發Android都是使用的eclipse,最近由於和外國朋友Timothy一起開發一款應用,他是從WP平台剛切換使用Android的,使用的開發環境時Android
Android本身已經提供了ProgressDialog進度等待框,使用該Dialog,我們可以為用戶提供更好的體驗:在網絡請求時,彈出此框等待網絡數據。 不過,既然是為