Python 类
定义类 在写你的第一个类之前,你应该知道它的语法。我们以下面这种方式定义类:
1 2 3 4 class nameoftheclass(parent_class): statement1 statement2 statement3
在类的声明中你可以写任何 Python 语句,包括定义函数(在类中我们称为方法)。
1 2 3 4 5 >>> class MyClass(object): ... """A simple example class""" ... i = 12345 ... def f(self): ... return 'hello world'
init 方法类的实例化使用函数符号。只要将类对象看作是一个返回新的类实例的无参数函数即可。例如(假设沿用前面的类):
以上创建了一个新的类实例并将该对象赋给局部变量 x。
这个实例化操作创建一个空的对象。很多类都倾向于将对象创建为有初始状态的。因此类可能会定义一个名为 init () 的特殊方法,像下面这样:
1 2 def __init__(self): self.data = []
类定义了 init () 方法的话,类的实例化操作会自动为新创建的类实例调用 init () 方法。所以在下例中,可以这样创建一个新的实例:
当然,出于弹性的需要,init () 方法可以有参数。事实上,参数通过init () 传递到类的实例化操作上。例如:
1 2 3 4 5 6 7 8 >>> class Complex: ... def __init__(self, realpart, imagpart): ... self.r = realpart ... self.i = imagpart ... >>> x = Complex(3.0, -4.5) >>> x.r, x.i (3.0, -4.5)
继承 当一个类继承另一个类时,它将继承父类的所有功能(如变量和方法)。这有助于重用代码。
在下一个例子中我们首先创建一个叫做 Person 的类,然后创建两个派生类 Student 和 Teacher。当两个类都从 Person 类继承时,它们的类除了会有 Person 类的所有方法还会有自身用途的新方法和新变量。
student_teacher.py 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 #!/usr/bin/env python3 class Person(object): """ 返回具有给定名称的 Person 对象 """ def __init__(self, name): self.name = name def get_details(self): """ 返回包含人名的字符串 """ return self.name class Student(Person): """ 返回 Student 对象,采用 name, branch, year 3 个参数 """ def __init__(self, name, branch, year): Person.__init__(self, name) self.branch = branch self.year = year def get_details(self): """ 返回包含学生具体信息的字符串 """ return "{} studies {} and is in {} year.".format(self.name, self.branch, self.year) class Teacher(Person): """ 返回 Teacher 对象,采用字符串列表作为参数 """ def __init__(self, name, papers): Person.__init__(self, name) self.papers = papers def get_details(self): return "{} teaches {}".format(self.name, ','.join(self.papers)) person1 = Person('Sachin') student1 = Student('Kushal', 'CSE', 2005) teacher1 = Teacher('Prashad', ['C', 'C++']) print(person1.get_details()) print(student1.get_details()) print(teacher1.get_details())
在这个例子中你能看到我们是怎样在 Student 类和 Teacher 类中调用 Person 类的 init 方法。
我们也在 Student 类和 Teacher 类中重写了 Person 类的 get_details() 方法。
因此,当我们调用 student1 和 teacher1 的 get_details() 方法时,使用的是各自类(Student 和 Teacher)中定义的方法。
多继承 一个类可以继承自多个类,具有父类的所有变量和方法,语法如下:
1 2 3 4 5 6 class MyClass(Parentclass1, Parentclass2,...): def __init__(self): Parentclass1.__init__(self) Parentclass2.__init__(self) ... ...
删除对象 现在我们已经知道怎样创建对象,现在我们来看看怎样删除一个对象。我们使用关键字 del 来做到这个。
1 2 3 4 5 6 >>> s = "I love you" >>> del s >>> s Traceback (most recent call last): File "<stdin>", line 1, in <module> NameError: name 's' is not defined
del 实际上使对象的引用计数减少一,当对象的引用计数变成零的时候,垃圾回收器会删除这个对象。
属性(attributes)读取方法 在 Python 里请不要使用属性(attributes)读取方法(getters 和 setters)。如果你之前学过其它语言(比如 Java),你可能会想要在你的类里面定义属性读取方法。请不要这样做,直接使用属性就可以了,就像下面这样:
1 2 3 4 5 6 7 8 9 10 >>> class Student(object): ... def __init__(self, name): ... self.name = name ... >>> std = Student("Kushal Das") >>> print(std.name) Kushal Das >>> std.name = "Python" >>> print(std.name) Python
Properties 装饰器 你可能想要更精确的调整控制属性访问权限,你可以使用 @property 装饰器,@property 装饰器就是负责把一个方法变成属性调用的。
下面有个银行账号的例子,我们要确保没人能设置金额为负,并且有个只读属性 cny 返回换算人名币后的金额。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 #!/usr/bin/env python3 class Account(object): """账号类, amount 是美元金额. """ def __init__(self, rate): self.__amt = 0 self.rate = rate @property def amount(self): """账号余额(美元)""" return self.__amt @property def cny(self): """账号余额(人名币)""" return self.__amt * self.rate @amount.setter def amount(self, value): if value < 0: print("Sorry, no negative amount in the account.") return # return 的作用:当输参数不符合条件,立即退出函数而不执行下面的赋值操作。 self.__amt = value if __name__ == '__main__': acc = Account(rate=6.6) # 基于课程编写时的汇率 acc.amount = 20 print("Dollar amount:", acc.amount) print("In CNY:", acc.cny) acc.amount = -100 print("Dollar amount:", acc.amount)
运行程序: