深入了解Android自定義屬性

自定義view的時(shí)候号醉,有時(shí)需要用到自定義屬性苍苞,方便我們定制View矾飞。一般來說挥萌,自定義屬性過程如下:

  1. 定義屬性:在values下的attrs.xml內(nèi)編寫declare-styleable標(biāo)簽來定義屬性;
  2. 使用屬性:在布局文件中通過獲取
  3. 獲取屬性:在自定義view中使用TypedArray獲取自定義的屬性亡问。

下面按照自定義View的流程講講自定義屬性:
首先說明一下自定義View的四個(gè)構(gòu)造函數(shù)的區(qū)別:

public class CustomView extends View {
    public CustomView(Context context) {
        super(context);
    }

    public CustomView(Context context, @Nullable AttributeSet attrs) {
        super(context, attrs);
    }

    public CustomView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
    }

    public CustomView(Context context, @Nullable AttributeSet attrs, int defStyleAttr, int defStyleRes) {
        super(context, attrs, defStyleAttr, defStyleRes);
    }
}

在使用過程中官紫,一般都是聯(lián)級(jí)調(diào)用。第一個(gè)構(gòu)造函數(shù)用處并不大州藕,主要在Java代碼中聲明View時(shí)才使用束世;在布局文件中使用自定義的View,則調(diào)用的是第二個(gè)構(gòu)造函數(shù)床玻;第三毁涉,第四個(gè)構(gòu)造函數(shù)是與系統(tǒng)主題有關(guān)的,從參數(shù)defStyleAttrdefStyleRes也可以看得出來锈死;也就是說贫堰,在自定義View時(shí),如果不需要view跟隨主題改變待牵,則前兩個(gè)構(gòu)造函數(shù)就足夠了其屏。

關(guān)于AttributeSet

在第二個(gè)構(gòu)造函數(shù)中,有個(gè)AttributeSet參數(shù)缨该,那這個(gè)參數(shù)有啥用的呢偎行?
查看源碼,我們可以發(fā)現(xiàn)AttributeSet是個(gè)接口,該接口用于解析xml布局中的屬性蛤袒。舉個(gè)例子熄云,假設(shè)定義了一個(gè)CustomerView,在布局中使用:

<com.example.developmc.customview.CustomView
        android:layout_width="100dp"
        android:layout_height="100dp" />

然后我們?cè)跇?gòu)造函數(shù)中用如下代碼打印AttributeSet中的屬性:

public CustomView(Context context, @Nullable AttributeSet attrs) {
        super(context, attrs);
        int attributeCount = attrs.getAttributeCount();
        for (int i = 0; i < attributeCount; i++) {
            String attrName = attrs.getAttributeName(i);
            String attrVal = attrs.getAttributeValue(i);
            Log.e("AttributeSet:", "attrName = " + attrName + " , attrVal = " + attrVal);
        }
    }

打印結(jié)果如下:

attrName = layout_width , attrVal = 100.0dip
attrName = layout_height , attrVal = 100.0dip

可以看到AttributeSet包含了所有在布局中定義的屬性,并且能夠按順序地取得各個(gè)屬性的name和value妙真。換言之缴允,AttributeSet用于解析View在xml布局中的所有屬性的name和value,這也是自定義View要使用第二個(gè)構(gòu)造函數(shù)的原因珍德。

細(xì)心觀察一下练般,我們?cè)诓季种?strong>layout_width的值是100dp,但打印出來的值卻是100dip,這單位不對(duì)呀菱阵!抱著疑問踢俄,我們修改一下布局:

<com.example.developmc.customview.CustomView
        android:layout_width="@dimen/dimen_100"
        android:layout_height="100dp" />

其中dimen_100的定義是:

<dimen name="dimen_100">100dp</dimen>

再次打印,結(jié)果如下:

attrName = layout_width , attrVal = @2131165262
attrName = layout_height , attrVal = 100.0dip

可以看到layout_width變成了一個(gè)奇怪的值@2131165262晴及,這個(gè)值是怎么來的呢都办?
我們知道,Android會(huì)在R.java中為定義的屬性生成資源標(biāo)識(shí)符(一個(gè)十六進(jìn)制的數(shù)值)虑稼。在app/build/generated/r/debug下找到并打開R.java琳钉,找到dimen_100,對(duì)應(yīng)的值是0x7f07004e蛛倦,將這個(gè)數(shù)值轉(zhuǎn)為十進(jìn)制歌懒,正好就是“2131165262”!

到這里溯壶,我們可以得出結(jié)論及皂,當(dāng)在布局中直接賦值(如:android:layout_width="100dp"),Attribute拿到的值且改,數(shù)值是正確的验烧,但單位可能會(huì)不對(duì); 當(dāng)在布局中為屬性賦引用值(如:android:layout_width="@dimen/dimen_100")又跛,Attribute拿到的是該值對(duì)應(yīng)的資源標(biāo)識(shí)符碍拆!總的來說,Attribute解析出來的屬性值并不能直接使用慨蓝!不要怕感混,TypedArray就是用于簡(jiǎn)化這方面的工作的。

關(guān)于TypedArray

先來回憶一下礼烈,我們是如何同時(shí)使用Arrtibute和TypedArray的:
首先弧满,新建文件attrs.xml,為自定義View添加一個(gè)自定義屬性:

<declare-styleable name="CustomView">
        <attr name="customWidth" format="dimension"/>
    </declare-styleable>

然后在布局文件中使用自定義的屬性customWidth

<com.example.developmc.customview.CustomView
        android:layout_width="@dimen/dimen_100"
        android:layout_height="100dp"
        app:customWidth="@dimen/dimen_100"/>

最后在構(gòu)造函數(shù)中用TypedArray解析customWidth的value此熬,代碼如下:

public CustomView(Context context, @Nullable AttributeSet attrs) {
        super(context, attrs);
        int attributeCount = attrs.getAttributeCount();
        for (int i = 0; i < attributeCount; i++) {
            String attrName = attrs.getAttributeName(i);
            String attrVal = attrs.getAttributeValue(i);
            Log.e("AttributeSet:", "attrName = " + attrName + " , attrVal = " + attrVal);
        }
        TypedArray array = context.obtainStyledAttributes(attrs,R.styleable.CustomView);
        float width = array.getDimension(R.styleable.CustomView_customWidth,200f);
        Log.e("width:", "widthVal = "+String.valueOf(width));
        array.recycle();
    }

打印結(jié)果是:

attrName = layout_width , attrVal = @2131165262
attrName = layout_height , attrVal = 100.0dip
attrName = customWidth , attrVal = @2131165262
widthVal = 350.0

測(cè)試虛擬機(jī)的像素是:560dpi谱秽,那么通過計(jì)算洽蛀,100dp對(duì)應(yīng)的像素就是350dip
通過打印結(jié)果,可以看到使用Attribute和TypedArray的區(qū)別:如果使用Attribute疟赊,拿到的結(jié)果并不能直接使用,需要進(jìn)一步處理峡碉;而TypedArray則直接取得了正確的數(shù)值近哟,簡(jiǎn)化了這個(gè)步驟。所以在使用自定義屬性時(shí)鲫寄,我們總是應(yīng)該使用TypedArray的方式獲取屬性值吉执。
這里需要注意的一點(diǎn)是:每次使用完TypedArray之后,要記得調(diào)用recycle()回收地来。這是為什么呢戳玫?
從上述代碼我們是通過context.obtainStyledAttributes獲取TypeadArray實(shí)例的,并不是通過new實(shí)例的方式獲取的未斑。事實(shí)上咕宿,TypedArray類,沒有公有的構(gòu)造函數(shù)蜡秽,是一個(gè)典型的單例模式府阀,程序在運(yùn)行時(shí)維護(hù)一個(gè)TypedArray池,使用時(shí)芽突,向該池中請(qǐng)求一個(gè)實(shí)例试浙,用完之后,調(diào)用 recycle() 方法來釋放該實(shí)例寞蚌,從而使其可被其他模塊復(fù)用田巴。

關(guān)于declare-styleable

我們一般在attrs.xml中通過<declare-styleable>標(biāo)簽聲明自定義的屬性;先看看下面的代碼:

<resources>
    <declare-styleable name="CustomView">
        <attr name="customWidth" format="dimension"/>
    </declare-styleable>
    <attr name="customHeight" format="dimension"/>
</resources>

在上述attrs.xml中挟秤,我們自定義了兩個(gè)屬性:customWidth和customHeight壹哺,其中customHeight沒有聲明在declare-styleable。運(yùn)行代碼后煞聪,在生成的R.java文件中斗躏,可以同時(shí)看到這兩個(gè)屬性:

public static final class attr {
    public static final int customWidth=0x7f0100ce;
    public static final int customHeight=0x7f010001;
}

可以看到定義在declare-styleable中與直接用attr定義沒有實(shí)質(zhì)的不同,系統(tǒng)都會(huì)為我們?cè)赗.attr中生成響應(yīng)的屬性昔脯。但不同的是啄糙,如果聲明在declare-styleable中,系統(tǒng)除了在R.java的attr類下生成資源標(biāo)識(shí)符云稚,還會(huì)為我們?cè)赗.java內(nèi)的styleable類中生成相關(guān)的屬性:

public static final class styleable {
      public static final int[] CustomView = {
            0x7f0100ce
        };
}

可以看到customWidth屬性對(duì)應(yīng)的標(biāo)識(shí)符0x7f0100ce被保存在styleable內(nèi)的數(shù)組中隧饼。如上所示,R.styleable.CustomView是一個(gè)int[]静陈,而里面的元素正是declare-styleable內(nèi)聲明的元素燕雁,這個(gè)數(shù)組在自定義View的構(gòu)造函數(shù)中獲得屬性值時(shí)會(huì)用到诞丽。
將自定義屬性分組聲明在declare-styleabe中的作用就是系統(tǒng)會(huì)自動(dòng)為我們生成這些東西,如果不聲明在declare-styleable中拐格,我們也可以在需要的時(shí)候自己構(gòu)建這個(gè)數(shù)組僧免,只是比較麻煩。當(dāng)我們有多個(gè)自定義View需要用到同一個(gè)自定義屬性時(shí)捏浊,就不能同時(shí)在兩個(gè)declare-styleabe下聲明同一個(gè)屬性了(編譯不通過)懂衩,這時(shí)就可以把這個(gè)屬性直接定義在attr下了,然后在需要使用時(shí)金踪,自己構(gòu)建數(shù)組引用即可浊洞。
本文就到這里,謝謝各位胡岔。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末法希,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子靶瘸,更是在濱河造成了極大的恐慌苫亦,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,123評(píng)論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件奕锌,死亡現(xiàn)場(chǎng)離奇詭異著觉,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)惊暴,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,031評(píng)論 2 384
  • 文/潘曉璐 我一進(jìn)店門饼丘,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人辽话,你說我怎么就攤上這事肄鸽。” “怎么了油啤?”我有些...
    開封第一講書人閱讀 156,723評(píng)論 0 345
  • 文/不壞的土叔 我叫張陵典徘,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我益咬,道長(zhǎng)逮诲,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,357評(píng)論 1 283
  • 正文 為了忘掉前任幽告,我火速辦了婚禮梅鹦,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘冗锁。我一直安慰自己齐唆,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,412評(píng)論 5 384
  • 文/花漫 我一把揭開白布冻河。 她就那樣靜靜地躺著箍邮,像睡著了一般茉帅。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上锭弊,一...
    開封第一講書人閱讀 49,760評(píng)論 1 289
  • 那天堪澎,我揣著相機(jī)與錄音,去河邊找鬼廷蓉。 笑死全封,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的桃犬。 我是一名探鬼主播,決...
    沈念sama閱讀 38,904評(píng)論 3 405
  • 文/蒼蘭香墨 我猛地睜開眼行楞,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼攒暇!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起子房,我...
    開封第一講書人閱讀 37,672評(píng)論 0 266
  • 序言:老撾萬榮一對(duì)情侶失蹤形用,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后证杭,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體田度,經(jīng)...
    沈念sama閱讀 44,118評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,456評(píng)論 2 325
  • 正文 我和宋清朗相戀三年解愤,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了镇饺。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,599評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡送讲,死狀恐怖奸笤,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情哼鬓,我是刑警寧澤监右,帶...
    沈念sama閱讀 34,264評(píng)論 4 328
  • 正文 年R本政府宣布,位于F島的核電站异希,受9級(jí)特大地震影響健盒,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜称簿,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,857評(píng)論 3 312
  • 文/蒙蒙 一扣癣、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧予跌,春花似錦搏色、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,731評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽垂涯。三九已至,卻和暖如春航邢,著一層夾襖步出監(jiān)牢的瞬間耕赘,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,956評(píng)論 1 264
  • 我被黑心中介騙來泰國打工膳殷, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留操骡,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,286評(píng)論 2 360
  • 正文 我出身青樓赚窃,卻偏偏與公主長(zhǎng)得像册招,于是被迫代替她去往敵國和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子勒极,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,465評(píng)論 2 348

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