Android 深入理解Android中的自定義屬性

發表於2015-05-11

1、引言

對於自定義屬性,大家肯定都不陌生,遵循以下幾步,就可以實現:

  1. 自定義一個CustomView(extends View )類
  2. 編寫values/attrs.xml,在其中編寫styleable和item等標籤元素
  3. 在佈局檔案中CustomView使用自定義的屬性(注意namespace)
  4. 在CustomView的構造方法中通過TypedArray獲取

ps:如果你對上述幾個步驟不熟悉,建議先熟悉下,再繼續~

那麼,我有幾個問題:

  • 以上步驟是如何奏效的?
  • styleable 的含義是什麼?可以不寫嘛?我自定義屬性,我宣告屬性就好了,為什麼一定要寫個styleable呢?
  • 如果系統中已經有了語義比較明確的屬性,我可以直接使用嘛?
  • 構造方法中的有個引數叫做AttributeSet
    (eg: MyTextView(Context context, AttributeSet attrs) )這個引數看名字就知道包含的是引數的陣列,那麼我能不能通過它去獲取我的自定義屬性呢?
  • TypedArray是什麼鬼?從哪冒出來的,就要我去使用?

恩,針對這幾個問題,大家可以考慮下,如何回答呢?還是說:老子會背上述4個步驟就夠了~~

2、常見的例子

接下來通過例子來回答上述問題,問題的回答順序不定~~大家先看一個常見的例子,即上述幾個步驟的程式碼化。

  • 自定義屬性的宣告檔案

  • 自定義View類

  • 佈局檔案中使用

ok,大家花3s掃一下,執行結果為:

應該都不意外吧,注意下,我的styleable的name寫的是test,所以說這裡並不要求一定是自定義View的名字。

3、AttributeSet與TypedArray

下面考慮:

構造方法中的有個引數叫做AttributeSet(eg: MyTextView(Context context, AttributeSet attrs) )這個引數看名字就知道包含的是引數的集合,那麼我能不能通過它去獲取我的自定義屬性呢?

首先AttributeSet中的確儲存的是該View宣告的所有的屬性,並且外面的確可以通過它去獲取(自定義的)屬性,怎麼做呢?
其實看下AttributeSet的方法就明白了,下面看程式碼。

輸出:

結合上面的佈局檔案,你發現了什麼?
我擦,果然很神奇,真的獲得所有的屬性,恩,沒錯,通過AttributeSet可以獲得佈局檔案中定義的所有屬性的key和value(還有一些方法,自己去嘗試),那麼是不是說TypedArray這個鬼可以拋棄了呢?答案是:NO!

現在關注下一個問題:

TypedArray是什麼鬼?從哪冒出來的,就要我去使用?

我們簡單修改下,佈局檔案中的MyTextView的屬性。

現在再次執行的結果是:

發現了什麼?通過AttributeSet獲取的值,如果是引用都變成了@+數字的字串。你說,這玩意你能看懂麼?那麼你看看最後一行使用TypedArray獲取的值,是不是瞬間明白了什麼。

TypedArray其實是用來簡化我們的工作的,比如上例,如果佈局中的屬性的值是引用型別(比如:@dimen/dp100),如果使用AttributeSet去獲得最終的畫素值,那麼需要第一步拿到id,第二步再去解析id。而TypedArray正是幫我們簡化了這個過程。

貼一下:如果通過AttributeSet獲取最終的畫素值的過程:

ok,現在別人問你TypedArray存在的意義,你就可以告訴他了。

4、declare-styleable

我們已經解決了兩個問題,接下來,我們看看佈局檔案,我們有一個屬性叫做:zhy:text
總所周知,系統提供了一個屬性叫做:android:text,那麼我覺得直接使用android:text更nice,這樣的話,考慮問題:

如果系統中已經有了語義比較明確的屬性,我可以直接使用嘛?

答案是可以的,怎麼做呢?
直接在attrs.xml中使用android:text屬性。

注意,這裡我們是使用已經定義好的屬性,不需要去新增format屬性(注意宣告和使用的區別,差別就是有沒有format)。
然後在類中這麼獲取:ta.getString(R.styleable.test_android_text);佈局檔案中直接android:text="@string/hello_world"即可。

這裡提一下,系統中定義的屬性,其實和我們自定義屬性的方式類似,你可以在sdk/platforms/android-xx/data/res/values該目錄下看到系統中定義的屬性。然後你可以在系統提供的View(eg:TextView)的構造方法中發現TypedArray獲取屬性的程式碼(自己去看一下)。

ok,接下來,我在想,既然declare-styleable這個標籤的name都能隨便寫,這麼隨意的話,那麼考慮問題:

styleable 的含義是什麼?可以不寫嘛?我自定義屬性,我宣告屬性就好了,為什麼一定要寫個styleable呢?

其實的確是可以不寫的,怎麼做呢?

  • 首先刪除declare-styleable的標籤

那麼現在的attrs.xml為:

喲西,so清爽~
* MyTextView實現

貌似多了些程式碼,可以看到我們宣告瞭一個int陣列,陣列中的元素就是我們想要獲取的attr的id。並且我們根據元素的在陣列中的位置,定義了一些整形的常量代表其下標,然後通過TypedArray進行獲取。
可以看到,我們原本的:

那麼其實呢?android在其內部也會這麼做,按照傳統的寫法,它會在R.java生成如下程式碼:

ok,根據上述你應該發現了什麼。styleale的出現系統可以為我們完成很多常量(int[]陣列,下標常量)等的編寫,簡化我們的開發工作(想想如果一堆屬性,自己編寫常量,你得寫成什麼樣的程式碼)。那麼大家肯定還知道declare-styleable的name屬性,一般情況下寫的都是我們自定義View的類名。主要為了直觀的表達,該declare-styleable的屬性,都是改View所用的。

其實瞭解該原理是有用的,詳見:Android 自定義控制元件 優雅實現元素間的分割線

ok,現在5個問題,回答了4個,第一個問題:

自定義屬性的幾個步驟是如何奏效的?

恩,上述以及基本涵蓋了這個問題的答案,大家自己總結,所以:略。

總結下今天的部落格。

  • attrs.xml裡面的declare-styleable以及item,android會根據其在R.java中生成一些常量方便我們使用(aapt乾的),本質上,我們可以不宣告declare-styleable僅僅宣告所需的屬性即可。
  • 我們在View的構造方法中,可以通過AttributeSet去獲得自定義屬性的值,但是比較麻煩,而TypedArray可以很方便的便於我們去獲取。
  • 我們在自定義View的時候,可以使用系統已經定義的屬性。

近期的更新計劃:自定義View的一些細節相關的Blog(重點會在互動上),Android最佳實踐相關的文章,framework相關的一些文章,敬請期待。

相關文章