亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 系統 > Android > 正文

理解Android中的自定義屬性

2020-04-11 10:53:33
字體:
來源:轉載
供稿:網友

本文實例講解了Android中的自定義屬性,具體內容如下

1、引言

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

  • 自定義一個CustomView(extends View )類
  • 編寫values/attrs.xml,在其中編寫styleable和item等標簽元素
  • 在布局文件中CustomView使用自定義的屬性(注意namespace)
  • 在CustomView的構造方法中通過TypedArray獲取

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

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

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

2、常見的例子

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

自定義屬性的聲明文件

  <?xml version="1.0" encoding="utf-8"?><resources>  <declare-styleable name="test">    <attr name="text" format="string" />    <attr name="testAttr" format="integer" />  </declare-styleable></resources>

自定義View類

package com.example.test;import android.content.Context;import android.content.res.TypedArray;import android.util.AttributeSet;import android.util.Log;import android.view.View;public class MyTextView extends View {  private static final String TAG = MyTextView.class.getSimpleName();  public MyTextView(Context context, AttributeSet attrs) {    super(context, attrs);    TypedArray ta = context.obtainStyledAttributes(attrs, R.styleable.test);    String text = ta.getString(R.styleable.test_testAttr);    int textAttr = ta.getInteger(R.styleable.test_text, -1);    Log.e(TAG, "text = " + text + " , textAttr = " + textAttr);    ta.recycle();  }}

布局文件中使用

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"  xmlns:tools="http://schemas.android.com/tools"  xmlns:zhy="http://schemas.android.com/apk/res/com.example.test"  android:layout_width="match_parent"  android:layout_height="match_parent" >  <com.example.test.MyTextView    android:layout_width="100dp"    android:layout_height="200dp"    zhy:testAttr="520"    zhy:text="helloworld" /></RelativeLayout>

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

 MyTextView: text = helloworld , textAttr = 520

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

3、AttributeSet與TypedArray

下面考慮:

構造方法中的有個參數叫做AttributeSet(eg: MyTextView(Context context, AttributeSet attrs) )這個參數看名字就知道包含的是參數的集合,那么我能不能通過它去獲取我的自定義屬性呢?
首先AttributeSet中的確保存的是該View聲明的所有的屬性,并且外面的確可以通過它去獲?。ㄗ远x的)屬性,怎么做呢?
其實看下AttributeSet的方法就明白了,下面看代碼。

public MyTextView(Context context, AttributeSet attrs) {    super(context, attrs);    int count = attrs.getAttributeCount();    for (int i = 0; i < count; i++) {      String attrName = attrs.getAttributeName(i);      String attrVal = attrs.getAttributeValue(i);      Log.e(TAG, "attrName = " + attrName + " , attrVal = " + attrVal);    }    // ==>use typedarray ...  }

輸出:

MyTextView(4136): attrName = layout_width , attrVal = 100.0dipMyTextView(4136): attrName = layout_height , attrVal = 200.0dipMyTextView(4136): attrName = text , attrVal = helloworldMyTextView(4136): attrName = testAttr , attrVal = 520

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

現在關注下一個問題:

TypedArray是什么鬼?從哪冒出來的,就要我去使用?
我們簡單修改下,布局文件中的MyTextView的屬性。

<com.example.test.MyTextView    android:layout_width="@dimen/dp100"    android:layout_height="@dimen/dp200"    zhy:testAttr="520"    zhy:text="@string/hello_world" />

現在再次運行的結果是:

MyTextView(4692): attrName = layout_width , attrVal = @2131165234MyTextView(4692): attrName = layout_height , attrVal = @2131165235MyTextView(4692): attrName = text , attrVal = @2131361809MyTextView(4692): attrName = testAttr , attrVal = 520>>use typedarrayMyTextView(4692): text = Hello world! , textAttr = 520

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

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

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

int widthDimensionId = attrs.getAttributeResourceValue(0, -1);    Log.e(TAG, "layout_width= "+getResources().getDimension(widthDimensionId));

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

4、declare-styleable

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

如果系統中已經有了語義比較明確的屬性,我可以直接使用嘛?
答案是可以的,怎么做呢?
直接在attrs.xml中使用android:text屬性。

<declare-styleable name="test">    <attr name="android:text" />    <attr name="testAttr" format="integer" />  </declare-styleable>

注意,這里我們是使用已經定義好的屬性,不需要去添加format屬性(注意聲明和使用的區別,差別就是有沒有format)。
然后在類中這么獲?。簍a.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為:

<?xml version="1.0" encoding="utf-8"?><resources>  <attr name="testAttr" format="integer" /></resources>

* MyTextView實現

package com.example.test;import android.content.Context;import android.content.res.TypedArray;import android.util.AttributeSet;import android.util.Log;import android.view.View;public class MyTextView extends View {  private static final String TAG = MyTextView.class.getSimpleName();  private static final int[] mAttr = { android.R.attr.text, R.attr.testAttr };  private static final int ATTR_ANDROID_TEXT = 0;  private static final int ATTR_TESTATTR = 1;  public MyTextView(Context context, AttributeSet attrs) {    super(context, attrs);    // ==>use typedarray    TypedArray ta = context.obtainStyledAttributes(attrs, mAttr);    String text = ta.getString(ATTR_ANDROID_TEXT);    int textAttr = ta.getInteger(ATTR_TESTATTR, -1);    //輸出 text = Hello world! , textAttr = 520    Log.e(TAG, "text = " + text + " , textAttr = " + textAttr);    ta.recycle();  }}

貌似多了些代碼,可以看到我們聲明了一個int數組,數組中的元素就是我們想要獲取的attr的id。并且我們根據元素的在數組中的位置,定義了一些整形的常量代表其下標,然后通過TypedArray進行獲取。
可以看到,我們原本的:

R.styleable.test => mAttrR.styleable.test_text => ATTR_ANDROID_TEXT(0)R.styleable.test_testAttr => ATTR_TESTATTR(1)

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

public static final class attr {  public static final int testAttr=0x7f0100a9;  }public static final class styleable {   public static final int test_android_text = 0;   public static final int test_testAttr = 1;   public static final int[] test = {      0x0101014f, 0x7f0100a9    };  }

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的時候,可以使用系統已經定義的屬性。

以上就是關于Android中的自定義屬性的相關內容,希望對大家的學習有所幫助。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲美女在线看| 欧美制服第一页| 亚洲国产另类 国产精品国产免费| 国产一区av在线| 精品福利在线看| 欧美噜噜久久久xxx| 91高清免费在线观看| 亚洲精品美女网站| 538国产精品一区二区在线| 91精品国产亚洲| 欧美自拍大量在线观看| 久久精品国产99国产精品澳门| 国产精品丝袜久久久久久高清| 韩日精品中文字幕| 久久久精品欧美| 欧美电影在线观看完整版| 欧美一区二区三区免费视| 国产精品久久久久久久久久久不卡| 欧美日韩国产区| 亚洲欧美中文在线视频| 国产精品福利小视频| 国产精品99久久久久久久久久久久| 韩剧1988在线观看免费完整版| 色综合导航网站| 欧美黄网免费在线观看| 最近中文字幕mv在线一区二区三区四区| 国产精品电影网| 久久久久久久影院| 欧美精品电影免费在线观看| 中文字幕久久久av一区| 日韩精品视频免费在线观看| 欧美wwwxxxx| 国产精品久久9| 91国内精品久久| 国产精品久久久久久久久影视| 78m国产成人精品视频| 欧美激情亚洲一区| 欧美孕妇与黑人孕交| 中文日韩电影网站| 日韩精品欧美国产精品忘忧草| 日韩电视剧免费观看网站| 亚洲日本中文字幕免费在线不卡| 久久久精品一区| 欧美大片第1页| 国产中文字幕亚洲| 亚洲精品视频久久| 国产91色在线| 色狠狠久久aa北条麻妃| 亚洲成人久久网| 欧美成人性色生活仑片| 久久免费精品日本久久中文字幕| 欧美日韩中文字幕综合视频| 国产一区二区三区丝袜| 中文字幕久久亚洲| 久久久精品影院| 成人性生交大片免费看小说| 成人免费视频a| 91热福利电影| 欧美性猛交丰臀xxxxx网站| 国产精品成人久久久久| 亚洲欧美三级伦理| 亚洲精品女av网站| 亚洲人成网站在线播| 国外色69视频在线观看| 久久激情视频免费观看| 亚洲成av人片在线观看香蕉| 久久人人爽国产| yw.139尤物在线精品视频| 青青久久aⅴ北条麻妃| 日韩最新av在线| 91精品久久久久久久久久久久久| 久久成人精品一区二区三区| 97久久超碰福利国产精品…| 国产在线视频2019最新视频| 上原亚衣av一区二区三区| 日韩三级成人av网| 国产色综合天天综合网| 欧美高清一级大片| 久操成人在线视频| 欧美精品videofree1080p| 国产日韩欧美黄色| 欧美一级视频在线观看| 色婷婷av一区二区三区在线观看| 欧美色道久久88综合亚洲精品| 亚洲午夜性刺激影院| 久久国产精品久久国产精品| 欧美午夜片欧美片在线观看| 国产拍精品一二三| 久久久久久久久久久成人| 日韩在线视频二区| 亚洲精品97久久| 久久精品国产精品亚洲| 91精品综合久久久久久五月天| 毛片精品免费在线观看| 国产精品一区二区av影院萌芽| 日本欧美爱爱爱| 欧美午夜性色大片在线观看| 麻豆国产va免费精品高清在线| 国产成人自拍视频在线观看| 亚洲国产精品va在线| 国产精品狠色婷| 国产精品夜间视频香蕉| 精品国偷自产在线视频| 91午夜在线播放| 国产mv免费观看入口亚洲| 成人网址在线观看| 欧美在线精品免播放器视频| 国产精品自产拍在线观| 91禁外国网站| 久久露脸国产精品| 亚洲欧美日韩国产精品| 午夜精品福利在线观看| 欧美日韩免费网站| 久久久视频免费观看| 欧美午夜女人视频在线| 国产香蕉一区二区三区在线视频| 国产精品亚洲第一区| 成人春色激情网| 蜜臀久久99精品久久久久久宅男| 欧美影院在线播放| 色爱av美腿丝袜综合粉嫩av| 91亚洲人电影| 久久综合久久美利坚合众国| 懂色av影视一区二区三区| 日韩欧美精品网址| 久久久999国产精品| 亚洲网址你懂得| 欧美激情国产日韩精品一区18| 亚洲大尺度美女在线| 一区二区三区动漫| 欧美日韩在线免费观看| 国产欧美精品日韩| 国产97在线|亚洲| 国产香蕉精品视频一区二区三区| 国产精品久久久久久久美男| 精品国产精品三级精品av网址| 欧美激情视频网站| 一区二区三区在线播放欧美| 国产亚洲视频在线| 在线成人激情黄色| 亚洲黄色av女优在线观看| 亚洲精品v天堂中文字幕| 黑人巨大精品欧美一区二区| 国产一区二区三区直播精品电影| 国产成人在线一区二区| 欧亚精品在线观看| 久久久精品2019中文字幕神马| 亚洲成人黄色网址| 国产亚洲欧洲高清一区| 久久综合国产精品台湾中文娱乐网| 欧美日韩一区二区三区在线免费观看| 欧美午夜精品久久久久久人妖| 国产精品自产拍在线观看| 色综合伊人色综合网| 国产精品电影观看| 亚洲人a成www在线影院| 欧美成人亚洲成人| 国产a∨精品一区二区三区不卡| 亚洲天堂av图片| 国产香蕉精品视频一区二区三区| 久久精品成人欧美大片| 日韩av在线看| www.久久久久| 亚洲人成绝费网站色www|