1、 用户定义类型

(1)用户定义的类型也称为类(class),类的定义形式如下:

class Point(object):

   """Represents a point in 2-D space."""

定义头表示新的类是一个Point,它是object的一种,而object是一个内置函数。

(2)定义一个叫做Point的类会创建一个类对象。

In [7]:print Point

<class '__main__.Point'>

 

  因为Point是在程序顶层定义的,它的“全名”是__main__.Point。

(3)类对象像一个创建对象的工厂。要新建一个Point对象,可以把Point当作函数来调用。

In [9]: blank=Point()

In [10]: print blank

<__main__.Pointobject at 0x000000000A121320>

返回值是到一个Point对象的引用,我们将它赋值给变量blank。新建一个对象的过程称为实例化(instantiation),而对象是这个类的一个实例。

当打印一个实例时,Python会告诉你它所属的类型,以及存储在内存中的位置,前缀0x表示后面的数字是16进制的。

 

2、属性

(1)      可以使用句点给实例赋值

In [21]: blank.x=3

In [22]: blank.y=4

注:x,y可以在类定义中不必事先定义

我们是将值赋值给一个对象的有命名的元素。这些元素称为属性(attribute)。

变量blank引用向一个Point对象,它包含两个属性。每个属性引用一个浮点数。可以使用相同的语法来读取一个属性的值。

In [26]: blank.x

Out[26]: 3

In [27]: blank.y

Out[27]: 4

(2) 可以在任意表达式中使用句点表示法。例如:

In [29]: print '(%g, %g)' %(blank.x, blank.y)

(3, 4)

In [30]: import math

In [31]: distance=math.sqrt(blank.x**2+blank.y**2)

In [32]: print distance

5.0

(3)可以将一个实例作为实参按照通常的方式传递

In [33]: def print_point(p):   

        """除‘%f’外,还可以用‘%e’、‘%g’表示浮点数,‘%e’用指数形式输出,‘%g’根据实际情况输出,默认情况下最多保留6位有效数字。"""

    ...: print '(%g,%g)'%(p.x,p.y)

    ...:

In [34]: print_point(blank)

(3,4)

另外,在函数中,p是blank的一个别名,如果函数修改了p,则blank也会改变,如下:

In [35]: def print_point(p):

    ...: print '(%g,%g)'%(p.x,p.y)

    ...: p.x=20

    ...:

In [36]: print_point(blank)

(3,4)

In [37]: blank.x

Out[37]: 20

(3) 补充(查看类的定义信息)

In [38]: help(Point)

Help on class Point in module __main__:

 

class Point(__builtin__.object)

| Represents a point in 2-D space.

|

| Data descriptors defined here:

|

| __dict__

| dictionary for instance variables (if defined)

|

| __weakref__

| list of weak references to theobject (if defined)

 

3、举例:设计一个表达矩形的类

有时候对象应该有哪些属性非常明显,有时候要根据需求做决定。

(1)定义一个类:

class Rectangle(object):
"""Represents a rectangle.
attributes: width, height, corner.
"""

文档字符串列出了属性:width和height是数字;corner是一个Point对象,用来指定左下角的顶点。

(2)要表达一个矩形,你需要实例化一个Rectangle对象,并对其属性赋值:

In [39]: class Rectangle(object):

    ...: """Represents a rectangle.

    ...:

    ...: attributes: width, height, corner.

    ...: """

    ...:

In [40]: box=Rectangle()

In [41]: box.width=100

In [42]: box.height=200

In [43]: box.corner=Point()

In [44]: box.corner.x=0

In [45]: box.corner.y=0

 表达式box.corner.x表示:去往box引用的对象,并选择属性corner,接着去往那个对象,并选择属性x.

 

4、作为返回值的实例

新建一个对象的过程称为实例化(instantiation),而对象是这个类的一个实例。

函数可以返回实例,例如:

In[48]: def find_center(rect):

    ...: p=Point()

    ...: p.x=rect.corner.x+rect.width/2.0

    ...: p.y=rect.corner.y+rect.height/2.0

    ...: return p

    ...:

函数可以返回实例。Find_center接收Rectangle对象作为参数,并返回一个point对象,包含这个Rectangle的中心点的坐标。例如:

In [49]: center=find_center(box)

In [50]: center.x

Out[50]: 50.0

In [51]: center.y

Out[51]: 100.0

In [54]: print_point(center)

(50,100)

 

5、对象是可变的

(1)可以通过对一个对象的某个属性赋值来修改它的状态,例如:要修改一个矩形的尺寸而保持它的位置不变,可以修改属性的width和height的值:

In [55]: box.width=box.width+50

In [56]: box.height=box.height+100

(2)也可以编写函数来修改对象,例如:

In [61]: def grow_rectangle(rect, dwidth, dheight):

    ...: rect.width=rect.width+dwidth

    ...: rect.height=rect.height+dheight

    ...:

In [62]: grow_rectangle(box,50,0)

In [63]: box.width

Out[63]: 250

In [64]: box.height

Out[64]: 300

In [65]: grow_rectangle(box,50,0)

 

In [66]: box.width

Out[66]: 300

In [67]: box.height

Out[67]: 300

在函数中,rect是box的别名,所以如果函数修改了rect,则box也改变了。

 

6、复制

(1)别名的使用有时候会让程序更难阅读,因为一个地方的修改可能会给其他地方带来意想不到的变化,要跟踪掌握所有引用到一个给定的对象的变量非常困难。

In [71]: box.width

Out[71]: 300

In [72]: box.height

Out[72]: 300

In [73]: box1=box

In [74]: box1.width

Out[74]: 300

In [75]: box1.height

Out[75]: 300

In [76]: box1.width=123

In [77]: box.width

Out[77]: 123

(2)浅复制

使用别名的替代方案是复制。copy模块里有一个函数copy可以复制任何对象:

In [78]: p1=Point()

In [79]: p1.x=3.0

In [80]: p1.y=4.0

In [81]: import copy

In [82]: p2=copy.copy(p1)

In [83]: print_point(p1)

(3,4)

In [84]: print_point(p2)

(3,4)

In [85]: p1 is p2

Out[85]: False

In [86]: p1==p2

Out[86]: False

is操作符判断出p1和p2不是同一个对象,==操作符的默认行为和is的操作符相同,它会检查唯一性,而不是对象的相等性。

如果使用copy.copy复制一个Retangle,你会发现它复制了Rectangle对象,但并不复制内嵌的Point对象。这种行为即混乱不清,又容易导致错误。

In [89]: box2=copy.copy(box)

In [90]: box2 is box

Out[90]: False

In [91]: box2.corner is box.corner

Out[91]: True

(4) 深复制(deep copy)推荐

针对copy.copy引起的问题,copy模块还提供了deepcopy方法,它不但复制对象,还复制对象中引用的对象,甚至他们引用的对象,依次类推。

In[92]: box3=copy.deepcopy(box)

In[93]: box3.corner is box.corner

Out[93]: False

 

7、调试

操作对象时,可能会遇到一些异常。如果访问一个不存在的对象,会得到AttributeError:

In [99]: p=Point()

In [100]: p.z

Traceback(most recent calllast):

File"<ipython-input-100-6dce4e43e15c>", line1,in<module>

p.z

AttributeError: 'Point' object has no attribute 'z'

(1) 如何不清除一个对象是什么类型,可以使用type

In [101]: type(p)

Out[101]: __main__.Point

(2) 不确定一个对象是否拥有某个特定属性,使用内置函数hasattr(has attr)查看:

In [102]: p.x=10

In [103]: hasattr(p,'x')

Out[103]: True

In [104]: hasattr(p,'y')

Out[104]: False

第一个形参可以是任何对象,第二个形参是一个字符串,包含属性的名称。

(3) 查看对象所有的属性dir

In[105]: dir(box)

Out[105]:

['__class__',

'__delattr__',

'__dict__',

'__doc__',

'__format__',

'__getattribute__',

'__hash__',

'__init__',

'__module__',

'__new__',

'__reduce__',

'__reduce_ex__',

'__repr__',

'__setattr__',

'__sizeof__',

'__slotnames__',

'__str__',

'__subclasshook__',

'__weakref__',

'corner',

'height',

'width']