深入理解Android中的自定义属性
本文出处:
http://blog.csdn.net/lmj623565791/article/details/45022631
本文出自:【张鸿洋的博客】
引言
对于自定义属性,大家肯定都不陌生,遵循以下几步,就可以实现:
- 自定义一个
CustomView(extends View)
类 - 编写
values/attrs.xml
,在其中编写styleable
和item
等标签元素 - 在布局文件中
CustomView
使用自定义的属性(注意namespace
) - 在
CustomView
的构造方法中通过TypedArray
获取
ps:如果你对上述几个步骤不熟悉,建议先熟悉下,再继续~
那么,我有几个问题:
- 以上步骤是如何奏效的?
styleable
的含义是什么?可以不写嘛?我自定义属性,我声明属性就好了,为什么一定要写个styleable
呢?- 如果系统中已经有了语义比较明确的属性,我可以直接使用嘛?
- 构造方法中的有个参数叫做
AttributeSet
(eg: MyTextView(Context context, AttributeSet attrs) )这个参数看名字就知道包含的是参数的数组,那么我能不能通过它去获取我的自定义属性呢? TypedArray
是什么鬼?从哪冒出来的,就要我去使用?
恩,针对这几个问题,大家可以考虑下,如何回答呢?还是说:老子会背上述4个步骤就够了~~
常见的例子
接下来通过例子来回答上述问题,问题的回答顺序不定~~大家先看一个常见的例子,即上述几个步骤的代码化。
- 自定义属性的声明文件
1 |
|
- 自定义View类
1 | package com.example.test; |
- 布局文件中使用
1 | <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" |
ok,大家花3s扫一下,运行结果为:
1 | MyTextView: text = helloworld , textAttr = 5201 |
应该都不意外吧,注意下,我的styleable
的name
写的是test
,所以说这里并不要求一定是自定义View
的名字。
AttributeSet
与TypedArray
下面考虑:
构造方法中的有个参数叫做
AttributeSet
(eg: MyTextView(Context context, AttributeSet attrs) )这个参数看名字就知道包含的是参数的集合,那么我能不能通过它去获取我的自定义属性呢?
首先AttributeSet
中的确保存的是该View声明的所有的属性,并且外面的确可以通过它去获取(自定义的)属性,怎么做呢?
其实看下AttributeSet
的方法就明白了,下面看代码。
1 | public MyTextView(Context context, AttributeSet attrs) { |
输出:
1 | MyTextView(4136): attrName = layout_width , attrVal = 100.0dip |
结合上面的布局文件,你发现了什么?
我擦,果然很神奇,真的获得所有的属性,恩,没错,通过AttributeSet
可以获得布局文件中定义的所有属性的key和value(还有一些方法,自己去尝试),那么是不是说TypedArray
这个鬼可以抛弃了呢?答案是:NO!
。
现在关注下一个问题:
TypedArray
是什么鬼?从哪冒出来的,就要我去使用?
我们简单修改下,布局文件中的MyTextView
的属性。
1 | <com.example.test.MyTextView |
现在再次运行的结果是:
1 | MyTextView(4692): attrName = layout_width , attrVal = @2131165234 |
发现了什么?通过AttributeSet
获取的值,如果是引用都变成了@+数字的字符串。你说,这玩意你能看懂么?那么你看看最后一行使用TypedArray获取的值,是不是瞬间明白了什么。
TypedArray
其实是用来简化我们的工作的,比如上例,如果布局中的属性的值是引用类型(比如:@dimen/dp100
),如果使用AttributeSet
去获得最终的像素值,那么需要第一步拿到id,第二步再去解析id。而TypedArray
正是帮我们简化了这个过程。
贴一下:如果通过AttributeSet
获取最终的像素值的过程:
1 | int widthDimensionId = attrs.getAttributeResourceValue(0, -1); |
ok,现在别人问你TypedArray
存在的意义,你就可以告诉他了。
declare-styleable
我们已经解决了两个问题,接下来,我们看看布局文件,我们有一个属性叫做:zhy:text
。
总所周知,系统提供了一个属性叫做:android:text
,那么我觉得直接使用android:text
更nice,这样的话,考虑问题:
如果系统中已经有了语义比较明确的属性,我可以直接使用嘛?
答案是可以的,怎么做呢?
直接在attrs.xml中使用android:text
属性。
1 | <declare-styleable name="test"> |
注意,这里我们是使用已经定义好的属性,不需要去添加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为:
1 |
|
哟西,so清爽~
* MyTextView实现
1 | package com.example.test; |
貌似多了些代码,可以看到我们声明了一个int数组,数组中的元素就是我们想要获取的attr的id。并且我们根据元素的在数组中的位置,定义了一些整形的常量代表其下标,然后通过TypedArray
进行获取。
可以看到,我们原本的:
1 | R.styleable.test => mAttr |
那么其实呢?android在其内部也会这么做,按照传统的写法,它会在R.java生成如下代码:
1 | public static final class attr { |
ok,根据上述你应该发现了什么。styleale的出现系统可以为我们完成很多常量(int[]数组,下标常量)等的编写,简化我们的开发工作(想想如果一堆属性,自己编写常量,你得写成什么样的代码)。那么大家肯定还知道declare-styleable
的name属性,一般情况下写的都是我们自定义View的类名。主要为了直观的表达,该declare-styleable
的属性,都是改View所用的。
其实了解该原理是有用的
ok,现在5个问题,回答了4个,第一个问题:
自定义属性的几个步骤是如何奏效的?
恩,上述以及基本涵盖了这个问题的答案,大家自己总结,所以:略。
总结下今天的博客。
attrs.xml
里面的declare-styleable
以及item,android会根据其在R.java中生成一些常量方便我们使用(aapt干的),本质上,我们可以不声明declare-styleable
仅仅声明所需的属性即可。- 我们在View的构造方法中,可以通过
AttributeSet
去获得自定义属性的值,但是比较麻烦,而TypedArray
可以很方便的便于我们去获取。 - 我们在自定义View的时候,可以使用系统已经定义的属性。