封装变量
廖雪峰:https://www.liaoxuefeng.com/wiki/1016959663602400/1017501655757856
当我们设置的属性的时候,直接抛出去给别人用,别人可以随便传值没有限制,当我们当限制属性值时,就不能直接抛出去了。
例如:
class Case:
def __int__(self,name,age):
self.name=name
self.age=age
c=Case()
c.name="sd"
c.age=222
显然,年龄不符合常理。所以我们使用设置属性方法来限制年龄。set_age(value)写入年龄方法,get_age()读取年龄方法。
# 封装
class Case:
def __int__(self,name,age):
self.name=name
#私有属性
# self.__age=age
self.set_age(age)
def get_age(self):
return self.__age
def set_age(self,value):
if value<100:
self.__age=value
else:
raise ValueError("输入年龄有误")
c=Case()
c.name="sd"
c.set_age(4232)
print(c.get_age())
print(c.__dict__)
进阶:上面的调用方法又略显复杂,没有直接用属性这么直接简单。所以需要使用装饰器。装饰器(decorator)可以给函数动态加上功能。Python内置的@property装饰器就是负责把一个方法变成属性调用的:
class Case:
def __int__(self,name,age):
self.name=name
#私有属性
self.age=age
# self.set_age(age)
@property#创建property,负责读取
def age(self):
return self.__age
@age.setter#写入
def age(self,value):
if value<100:
self.__age=value
else:
raise ValueError("输入年龄有误")
c=Case()
c.name="sd"
c.age=78
print(c.age)
__slots__用法
slots,定义在类中,作用是为了防止用户输入错误的变量名称。缺点丧失了动态语音给对象添加新的变量。
class Student(object):
pass
#创建对象
s = Student()
>>> s.name = 'Michael' # 动态给实例绑定一个属性
>>> print(s.name)
Michael
实例绑定一个方法:
>>> def set_age(self, age): # 定义一个函数作为实例方法
... self.age = age
...
>>> from types import MethodType
>>> s.set_age = MethodType(set_age, s) # 给实例绑定一个方法
>>> s.set_age(25) # 调用实例方法
>>> s.age # 测试结果
25
>>> s2 = Student() # 创建新的实例
>>> s2.set_age(25) # 尝试调用方法
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
AttributeError: 'Student' object has no attribute 'set_age'
为了给所有实例都绑定方法,可以给class绑定方法:
>>> def set_score(self, score):
... self.score = score
...
>>> Student.set_score = set_score
给class绑定方法后,所有实例均可调用:
>>> s.set_score(100)
>>> s.score
100
>>> s2.set_score(99)
>>> s2.score
99
使用__slots__
但是,如果我们想要限制实例的属性怎么办?比如,只允许对Student实例添加name和age属性。
为了达到限制的目的,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'
使用__slots__要注意,slots__定义的属性仅对当前类实例起作用,对继承的子类是不起作用的,除非在子类中也定义__slots,这样,子类实例允许定义的属性就是自身的__slots__加上父类的__slots__
>>> class GraduateStudent(Student):
... pass
...
>>> g = GraduateStudent()
>>> g.score = 9999