Android布局優(yōu)化之include双揪、ViewStub、merge

前言

在寫Android的xml布局時包帚,用好 include渔期、ViewStubmerge這三個標(biāo)簽,可以是我們的xml更加簡潔疯趟、高效拘哨。

include

按照官方的意思,include就是為了解決重復(fù)定義相同布局的問題信峻。
相當(dāng)于Java代碼中將相同的部分抽取出來倦青,然后復(fù)用,需要的時候引入它即可站欺,而不必每次都自己寫一遍姨夹。

舉例說明:

一個公共布局文件 my_layout.xml(這個布局后面例子也會用到):

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:id="@+id/linearLayout"
              android:layout_width="match_parent"
              android:layout_height="match_parent"
              android:orientation="vertical">
    <Button
        android:id="@+id/button"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="Button"/>
    <TextView
        android:id="@+id/textView"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="TextView"/>
</LinearLayout>

使用這個公共布局文件:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:layout_width="match_parent"
              android:layout_height="match_parent"
              android:orientation="vertical">

    <include
        android:id="@+id/include_layout"
        layout="@layout/my_layout"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"/>
</LinearLayout>
注意事項:

使用 include 最常見的問題就是 findViewById 時候出現(xiàn) NullPointerException
這個問題出現(xiàn)的前提是在 <include /> 中設(shè)置了id矾策,被 include 進(jìn)來的布局的根元素也設(shè)置了id磷账。
那么這時使用被 include 進(jìn)來的根元素id進(jìn)行 findViewById 就會出現(xiàn)NullPointerException

在上述例子中:

findViewById(R.id.include_layout);// 正常
findViewById(R.id.linearLayout);// 會出現(xiàn)NullPointerException
findViewById(R.id.button);// 正常
findViewById(R.id.textView); // 正常

ViewStub

ViewStub就是一個寬高都為0的一個View贾虽,它默認(rèn)是不可見的逃糟。
只有通過調(diào)用 setVisibility() 函數(shù)或者 Inflate() 函數(shù)才會將其要裝載的目標(biāo)布局給加載出來,從而達(dá)到延遲加載的效果蓬豁。
ViewStub布局可顯示之前绰咽,系統(tǒng)不會消耗資源去實例化里面的布局,可以節(jié)省系統(tǒng)資源消耗地粪。

設(shè)置 ViewStub 中延時加載的布局有兩種方式:

  1. 在xml中使用 android:layout 屬性來設(shè)置取募。
  2. 在代碼中使用 ViewStub.setLayoutResource(res); 來設(shè)置。

使ViewStub中布局顯示出來也有兩種方法:

  1. 調(diào)用 ViewStub.setVisibility(View.VISIBLE);蟆技。
  2. 調(diào)用 ViewStub.inflate();玩敏。

這兩個方法本質(zhì)上都是調(diào)用ViewStub.inflate();來實現(xiàn)布局的加載顯示。

舉例說明:
<ViewStub
    android:id="@+id/view_stub"
    android:layout_width="fill_parent"
    android:layout_height="49dp"
    android:layout="@layout/my_layout"
    android:inflatedId="@+id/view_stub_inflated_id"/>
// my_layout 布局文件就是上文中的 my_layout.xml 
// 第一種使用方法:
//使用android:layout="@layout/my_layout"設(shè)置布局
ViewStub viewStub = (ViewStub) findViewById(R.id.view_stub);  
//設(shè)置setVisibility质礼,使布局文件實例化
viewStub .setVisibility(View.VISIBLE);  
// 通過ViewStub的xml中的屬性 inflatedId 來獲取View
LinearLayout linearLayout =(LinearLayout) findViewById(R.id.view_stub_inflated_id);  
if ( viewStub.getVisibility() == View.VISIBLE ) {  
    // 已經(jīng)加載成功
}
// 第二種使用方法:
ViewStub viewStub = (ViewStub) findViewById(R.id.view_stub);  
viewStub.setLayoutResource(R.layout.my_layout);
//使用 inflate旺聚,使布局文件實例化
LinearLayout linearLayout= (LinearLayout)viewStub.inflate();
if (linearLayout!= null){ 
    //已經(jīng)加載成功
} 
 

merge

merge它可以刪減多余的層級,優(yōu)化UI眶蕉。
例如你的主布局文件是垂直的LinearLayout砰粹,這時使用includemy_layout.xml 引入進(jìn)來。
新布局也是垂直的LinearLayout造挽,那么這個新的LinearLayout就沒有任何意義了碱璃。使用的話反而增加反應(yīng)時間。這時可以使用<merge/>標(biāo)簽優(yōu)化刽宪。

merge 原理就是在解析xml時候厘贼,如果是 <merge/> 標(biāo)簽,那么直接將其中的子元素添加到merge 標(biāo)簽parent中圣拄,這樣就保證了不會引入額外的層級

示例如下 :

<?xml version="1.0" encoding="utf-8"?>
<merge xmlns:android="http://schemas.android.com/apk/res/android"> 
    <Button
        android:id="@+id/button"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="Button"/>
    <TextView
        android:id="@+id/textView"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="TextView"/>
</merge>
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末嘴秸,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌岳掐,老刑警劉巖凭疮,帶你破解...
    沈念sama閱讀 216,496評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異串述,居然都是意外死亡执解,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,407評論 3 392
  • 文/潘曉璐 我一進(jìn)店門纲酗,熙熙樓的掌柜王于貴愁眉苦臉地迎上來衰腌,“玉大人,你說我怎么就攤上這事觅赊∮胰铮” “怎么了?”我有些...
    開封第一講書人閱讀 162,632評論 0 353
  • 文/不壞的土叔 我叫張陵吮螺,是天一觀的道長饶囚。 經(jīng)常有香客問我,道長鸠补,這世上最難降的妖魔是什么萝风? 我笑而不...
    開封第一講書人閱讀 58,180評論 1 292
  • 正文 為了忘掉前任,我火速辦了婚禮紫岩,結(jié)果婚禮上规惰,老公的妹妹穿的比我還像新娘。我一直安慰自己泉蝌,他們只是感情好卿拴,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,198評論 6 388
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著梨与,像睡著了一般。 火紅的嫁衣襯著肌膚如雪文狱。 梳的紋絲不亂的頭發(fā)上粥鞋,一...
    開封第一講書人閱讀 51,165評論 1 299
  • 那天,我揣著相機(jī)與錄音瞄崇,去河邊找鬼呻粹。 笑死,一個胖子當(dāng)著我的面吹牛苏研,可吹牛的內(nèi)容都是我干的等浊。 我是一名探鬼主播,決...
    沈念sama閱讀 40,052評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼摹蘑,長吁一口氣:“原來是場噩夢啊……” “哼筹燕!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 38,910評論 0 274
  • 序言:老撾萬榮一對情侶失蹤撒踪,失蹤者是張志新(化名)和其女友劉穎过咬,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體制妄,經(jīng)...
    沈念sama閱讀 45,324評論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡掸绞,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,542評論 2 332
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了耕捞。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片衔掸。...
    茶點(diǎn)故事閱讀 39,711評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖俺抽,靈堂內(nèi)的尸體忽然破棺而出敞映,到底是詐尸還是另有隱情,我是刑警寧澤凌埂,帶...
    沈念sama閱讀 35,424評論 5 343
  • 正文 年R本政府宣布驱显,位于F島的核電站,受9級特大地震影響瞳抓,放射性物質(zhì)發(fā)生泄漏埃疫。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,017評論 3 326
  • 文/蒙蒙 一孩哑、第九天 我趴在偏房一處隱蔽的房頂上張望栓霜。 院中可真熱鬧,春花似錦横蜒、人聲如沸胳蛮。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,668評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽仅炊。三九已至,卻和暖如春澎蛛,著一層夾襖步出監(jiān)牢的瞬間抚垄,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,823評論 1 269
  • 我被黑心中介騙來泰國打工谋逻, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留呆馁,地道東北人。 一個月前我還...
    沈念sama閱讀 47,722評論 2 368
  • 正文 我出身青樓毁兆,卻偏偏與公主長得像浙滤,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子气堕,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,611評論 2 353

推薦閱讀更多精彩內(nèi)容