定义Traits
最后更新于:2022-04-01 11:15:18
# 定义Traits
在Python程序中按照下面的步骤使用Traits库:
1. 从 enthought.traits.api 中载入你所需要的对象
2. 定义你想使用的traits
3. 从HasTraits类继承一个新类,在其中使用你定义的traits声明trait属性
通常第2、3步是放在一起的,也就是说定义traits的同时定义trait属性,在本手册中的大部分例子都是采用这种方式:
```
from enthought.traits.api import HasTraits, Float
class Person(HasTraits):
weight = Float(50.0)
```
这段程序定义了一个从HasTraits类继承的Person类,在其内部声明了一个名为weight的trait属性,其类型为浮点数,初始值为50.0。trait属性像类的属性一样定义,像实例的属性一样使用。下面我们来看看如何使用trait属性:
```
>>> joe = Person()
>>> joe.weight
50.0
>>> joe.weight = 70.5
>>> joe.weight = 70
>>> joe.weight = "89"
Traceback (most recent call last):
File "...trait_handlers.py", line 175, in error value )
TraitError: The 'weight' trait of a Person instance must be a float,
but a value of '89' <type 'str'> was specified.
```
由于joe是Person类的实例,因此它有一个名为weight的trait属性,并且初始值为50.0。由于weight是使用Float声明的,我们能将浮点数赋值给它,由于整数可以不丢失精度的转换为浮点数,因此整数也可以赋值给它。然而,把浮点数赋值给整数trait属性将会产生错误。由于字符串无法转换为浮点数,因此赋值为字符串产生错误,错误的提示信息告诉我们它需要浮点数。
有时候我们希望trait属性能够自动的进行强制类型转换,这样我们就可以将字符串赋值给类型为float的trait属性,省去了手工转换的麻烦。这种强制类型转换的trait属性都用Casting trait声明,所有的Casting trait都是以 **C** 开头的:
```
from enthought.traits.api import HasTraits, CFloat
class Person(HasTraits):
cweight = CFloat(50.0)
```
```
>>> bill = Person()
>>> bill.cweight = "90"
>>> bill.cweight
90.0
>>> bill.cweight = "abc"
Traceback (most recent call last)
...
```
这段程序用CFloat声明了一个强制类型转换的trait属性cweight。我们可以将能转换为浮点数的字符串"90"赋值给它,但是 "abc" 这样的字符串赋值仍然会抛出异常。 我们可以想象CFloat的内部处理:它先将传入的值用内部函数float()进行强制类型转换,然后把结果赋值给trait属性。
我们也可以先单独定义一个traits,然后用它来声明多个类的多个trait属性,下面是一个例子:
```
from enthought.traits.api import HasTraits, Range
coefficient = Range(-1.0, 1.0, 0.0)
class quadratic(HasTraits):
c2 = coefficient
c1 = coefficient
c0 = coefficient
x = Range(-100.0, 100.0, 0.0)
```
在这个例子中,我们需要定义多个trait属性,其类型都为Range(具有取值范围的浮点值),并且范围都是-1.0到1.0,初始值为0.0。为了体现代码重用,我们先用coefficient = Range(-1.0, 1.0, 0.0)定义了一个traits,然后在quadratic类中使用它定义三个trait属性:c0, c1, c2。
## 预定义的Traits
Traits库为Python的许多数据类型提供了预定义的trait类型。HasTraits派生的类中用trait类型名直接定义trait属性,这个类的所有实例都将拥有一个初始化为缺省值的属性,例如:
```
class Person(HasTraits):
age = Float
```
上面的例子为Person类定义了一个age属性,其类型为浮点数,并且被初始化为0.0(Float的缺省值)。如果你希望用别的值初始化trait属性的话,可以把这个值当作参数传递给trait类型:
```
age = Float(10.0)
```
几乎所有的trait类型都是可以带括号调用的,它可以接受缺省值或者其它的参数;还可以通过关键字参数接受元数据。 .. TODO:: 插入元数据链接
### 简单类型
对于每个Python的简单数据类型都对应两种trait类型:强制类型和自动转换类型。它们的区别在于:
* **强制型Trait** : 当这样trait属性被赋值为类型不匹配的数据时,会产生错误
* **自动型Trait** : 类型不匹配时会自动调用此类型对应的Pyton内置的转换函数进行类型转换
| 强制型Trait | 自动型Trait | Python对应的数据类型 | 内置缺省值 | 自动转换函数 |
| --- | --- | --- | --- | --- |
| Bool | CBool | Boolean | False | bool() |
| Complex | CComplex | Complex number | 0+0j | complex() |
| Float | CFloat | Floating point number | 0.0 | float() |
| Int | CInt | Plain integer | 0 | int() |
| Long | CLong | Long integer | 0L | int() |
| Str | CStr | String | '' | str() |
| Unicode | CUnicode | Unicode | u'' | unicode() |
下面的例子演示了强制型Trait和自动型Trait之间的区别:
```
>>> from enthought.traits.api import HasTraits, Float, CFloat
>>> class Person ( HasTraits ):
... weight = Float
... cweight = CFloat
>>> bill = Person()
>>> bill.weight = 180 # OK, 整数和浮点数匹配(转换为浮点数而不丢失信息)
>>> bill.cweight = 180 # OK,
>>> bill.weight = '180' # Error, 字符串和浮点数不匹配
>>> bill.cweight = '180' # OK, 调用float('180')转换为浮点
>>> print bill.cweight
180.0
```
### 其它类型
除了简单类型以外,Traits库还定义了许多其他的常用的数据类型。几乎所有的Trait类型都可以直接使用其名称或者当作函数调用,并且可以接受多种参数。
* **Any** : 任何对象;
```
Array( [dtype = None, shape = None, value = None, typecode = None, **metadata] )
```
* **Button** : 按钮类型,通常用来触发事件,参数都是用来描述界面中的按钮的样式;
```
Callable( [value = None, **metadata] )
```
* **CArray** : 可自动转换类型的numpy数组; 调用的参数和Array相同
* **Class** : Python老式类;
```
Code( [value = "", minlen = 0, maxlen = sys.maxint, regex = "", **metadata] )
```
* **Color** : 界面库中所采用的颜色对象;
```
CSet( [trait = None, value = None, items = True, **metadata] )
```
* **Constant** : 常量对象,其值不能改变,必须指定初始值;
```
Dict( [key_trait = None, value_trait = None, value = None, items = True, **metadata] )
```
* **Directory** : 表示某个目录的路径的字符串;
```
Either( val1*[, *val2, ..., valN, **metadata] )
```
* **Enum** : 枚举数据,其值可以是候选值中的一个;
```
Event( [trait = None, **metadata] )
```
* **Expression** : Python的表达式对象;
```
File( [value = "", filter = None, auto_set = False, entries = 10, exists = False, **metadata ] )
```
* **Font** : 界面库中表示字体的对象;
```
class Employee(HasTraits):
manager = self
```
定义了一个Employee类,它有一个manager属性,其类型为Employee,缺省值为对象本身:
```
>>> e = Employee()
>>> e.manager
<__main__.Employee object at 0x05DB72A0>
>>> e
<__main__.Employee object at 0x05DB72A0>
```
如果用This定义的话,那么缺省值为None。
一般来说,属性为某个类的实例时可以这样定义:
```
manager = Instance(Empolyee)
```
但是对于这个例子中,在定义manager属性时,Empolyee还不存在,因此无法如此定义。如果你喜欢这种方式的话,可以用Instance("Empolyee")来定义,当两个类的属性交叉引用时,可以使用这种字符串的方式来定义。
This和self不但可以表示类本身,还可以表示派生的类:
```
>>> from enthought.traits.api import HasTraits, This
>>> class Employee(HasTraits):
... manager = This
...
>>> class Executive(Employee):
... pass
...
>>> fred = Employee()
>>> mary = Executive()
>>> fred.manager = mary
>>> mary.manager = fred
```
#### 列出可能的值
使用Enum可以定义枚举类型,在Enum的定义中给出所有可能的值,这些值必须是Python的简单数据类型,例如字符串、整数、浮点数等等,各个可能的值的类型可以不一样。可以直接将可能的值作为参数,或者将其包在某个list中,第一个值为缺省值:
```
class Items(HasTraits):
count = Enum(None, 0, 1, 2, 3, "many")
# 或者:
# count = Enum([None, 0, 1, 2, 3, "many"])
```
下面是运行结果:
```
>>> item = Items()
>>> item.count = 2
>>> item.count = "many"
>>> item.count = 5
```
如果你希望候选值是可以变化的话,可以用values关键字指定定义侯选值的属性名:
```
class Items(HasTraits):
count_list = List([None, 0, 1, 2, 3, "many"])
count = Enum(values="count_list")
```
我们定义一个count_list列表,然后在Enum定义中用values关键字指定候选值为count_list属性。
```
>>> item = Items()
>>> item.count = 5
Traceback (most recent call last)
#... 略去错误提示,此错误提示无法显示侯选值列表
>>> item.count_list.append(5)
>>> item.count = 5
>>> item.count
5
```
## Trait的元数据
Trait对象可以有元数据属性,这些属性保存在HasTraits对象的trait字典中,为了解释什么是trait字典和元数据,让我们先来看一个例子:
```
from enthought.traits.api import *
class MetadataTest(HasTraits):
i = Int(99)
s = Str("test", desc="a string trait property")
test = MetadataTest()
```
在IPython中运行了上面的程序之后,我们对test进行如下操作:
> ```
> >>> test.traits()
> {'i': <enthought.traits.traits.CTrait object at 0x05D44EA0>,
> 'trait_added': <enthought.traits.traits.CTrait object at 0x05D17CE8>,
> 's': <enthought.traits.traits.CTrait object at 0x05D44EF8>,
> 'trait_modified': <enthought.traits.traits.CTrait object at 0x05D17C90>}
>
> ```
>
> ```
> >>> test.trait("i")
> <enthought.traits.traits.CTrait object at 0x05D44EA0>
>
> ```
>
> ```
> >>> test.trait("s").desc
> 'a string trait property'
>
> ```
通过调用HasTraits对象的traits方法可以得到一个包含其所有trait对象的字典。请注意,trait属性和trait对象是两个东西:
* **trait属性** : 用于保存实际的值,例如:test.i, test.s
* **trait对象** : 用于描述trait属性,例如:test.trait("i"), test.trait("s")
也就是说对于每一个trait属性都有一个与之对应的trait对象描述它。而元数据就是保存在trait对象中的额外的描述属性用的数据。我们看到test的trait对象除了i和s之外,还有trait_added和trait_modified,着两个在HasTraits类中定义。
元数据可以分为三类:
* **内部属性** : 这些属性是trait对象自带的,只读不能写
* **识别属性** : 这些属性是可以自由地设置的,它们可以改变trait的一些行为
* **任意属性** : 用户自己添加的属性,需要自己编写程序使用它们
### 内部元数据
下面的这些元数据属性在Traits库内部使用,用户可以读取它们的值。
* **array** : 是否是数组,不是数组的trait对象没有此属性
* **default** : 对应的trait属性的缺省值。也就是说: trait属性的缺省值是保存在与其对应的trait对象的元数据属性default中的:
```
>>> test.trait("i").default
99
```
* **default_kind** : 一个描述缺省值的类型的字符串,其值可以是 value, list, dict, self, factory, method等:
```
>>> test.trait("i").default_kind
'value'
```
* **inner_traits** : 内部的trait对象,在List, Dict等中使用,表示List和Dict内部对象的类型
* **trait_type** : 描述trait属性的数据类型的对象。下面的例子中,得到的就是定义trait属性时所用的Int类的对象:
```
>>> test.trait("i").trait_type
<enthought.traits.trait_types.Int object at 0x05DBD2D0>
```
* **type** : trait属性的分类,可以是constant, delegate, event, property, trait
```
>>> test.trait("i").type
'trait'
```
### 能识别的元数据
下面的元数据属性不是预定义的,但是可以被HasTraits对象使用:
* **desc** : 描述trait属性用的字符串,在界面中使用
* **editor** : 指定一个生成界面时用何种TraitEditor编辑对应的trait属性
* **label** : 界面中的trait属性编辑器的标签中的字符串
* **rich_compare** : 指定判断trait属性值发生变化的方式。True(缺省)表示按值比较;Flase表示按照对象指针比较
* **trait_value** : 指定trait属性是否接受TraitValu类的对象,缺省值为False。当它为True时,将trait属性设置为TraitValue(),将重置trait属性值为缺省值。
* **transient** : 指定当对象被保存(持久化)时是否保存此trait属性值。对于大多数trait属性来说,它的缺省值都是True。