Python中有一个很独特的功能,类属性可为实例属性提供默认值。下面Person类中有一个名为current_year的类属性。compute_age方法中用到了这个属性,而且都故意使用self.current_year读取它的值。因为Person本身没有current_year这个实例属性,所以self.current_year默认获取的是Person.current_year类属性的值。
class Person:
current_year = 2024
def __init__(self, name: str, birth_year: int):
self.name = name
self.birth_year = birth_year
def compute_age(self):
return self.current_year - self.birth_year
p = Person('cj', 2000)
age = p.compute_age()
print(age) # 24
但是,如果为不存在的实例属性赋值,那么将创建一个新的实例属性,接上例:
p.sex = 'woman'
print(p.sex) # 新的实例属性sex,输出woman
假如为current_year实例属性赋值,那么同名类属性不会受到影响。然而一旦这么做,实例读取的self.current_year是实例属性current_year,也就是把同名类属性给遮盖了。借助这个功能,可以为各个实例的current_year属性定制不同的值。
p.current_year = 2025
print(p.current_year) #设置实例属性current_year 2025
print(Person.current_year) #类属性current_year不受影响 2024
如果想修改类属性的值,必须直接在类上修改,不能通过实例修改。如果想修改所有实例的current_year属性的默认值,则可以像下面那样修改。
Person.current_year = 2026
print(Person.current_year) # 重新覆盖类属性,输出2026
print(p.current_year) #之前设置current_year实例属性,会遮盖类属性,因此还是输出2025
然而,有一种修改方法更为符合Python风格,而且效果持久,也更有针对性。类属性是公开的会被子类继承,于是我们经常会创建一个子类,只用于定制类的数据属性。Django基于类的视图就大量使用这种技术。
class SubPerson(Person):
current_year = 2027
sp = SubPerson('jay', 1999)
age = sp.compute_age()
print(age) # 输出28
print(SubPerson.current_year) # 类属性为2027,覆盖了超类的类属性current_price:2024
在类中覆盖类属性最好的方法就是继承父类,然后覆盖父类的类属性。