类和实例

1、类(class):有属性有方法就是类。下面是Student类:

class Student(object):
    def __init__(self, name, score):
        self.name = name
        self.score = score

    def print_score(self):
        print('%s: %s'%(self.name,self.score))

    def get_grade(self):
        if self.score >= 90:
            return 'A'
        elif self.score >= 60 :
            return 'B'
        else:
            return 'C'

2、实例(instance):具体的对象。下面是类Student创建的方法:

Vincent = Student('Vincent', 100)
Lisa = Student('Lisa', 80)

定义好了Student类,就可以用这个类创建具体的实例,如Vincent,Lisa。

创建实例的时候必须把相关的属性填写进去。通过定义__init__方法,创建实例的时候,就把name,score属性绑定上去。每次创建实例,Python会自动调用__init__方法,它也被成为构造函数。它的第一个参数永远是selfself不需要传入,Python会自动把实例变量传进去。

def __init__(self, name, score):
    self.name = name
    self.score = score
    
Vincent = Student('Vincent', 100)
Lisa = Student('Lisa', 80)

数据的封装

面向对象的另一个重要特点是数据的封装。在上面Student类的实例中,每个实例都自带namescore数据。我们可以通过函数来访问这些数据,比如打印成绩:

def print_score(self):
    print('%s: %s'%(self.name,self.score))

既然Student实例本身就拥有这些数据,要访问这些数据,就没有必要从外面的函数去访问,可以直接在Student类的内部定义访问数据的函数,这样,就把“数据”给封装起来了。这些封装数据的函数是和Student类本身是关联起来的,我们称之为类的方法。


使用__slots__

正常情况下,即使在定义一个class后,我们可以给对象绑定任何属性和方法。这是动态语言的灵活性。

class Student(object):
    pass

然后绑定属性name给Student

s= Student()
s.name = 'Michael'
print(s.name)
#结果是'Michael'

但是,如果我们要限制实例的属性怎么办?Python允许在定义class的时候,定义一个特殊的__slots__变量,来限制该class实例能添加的属性:

class Student(object):
    __slots__ = ('name', 'age') # 用tuple定义允许绑定的属性名称
    
s = Student() # 创建新的实例
s.name = 'Michael' # 绑定属性'name'
s.age = 25 # 绑定属性'age'
s.score = 99 # 绑定属性
'score'Traceback (most recent call last):
File "<stdin>", line 1, in <module>
AttributeError: 'Student' object has no attribute 'score'

由于'score'没有被放到__slots__中,所以不能绑定score属性,试图绑定score将得到AttributeError的错误。

使用__slots__要注意,__slots__定义的属性仅对当前类实例起作用,对继承的子类是不起作用的。