类属性
就像如下代码:
class Person:
name = "张三" # 共有类属性
__age = 18 # 私有类属性
在类中直接定义的属性就是类属性,它被所有的实例对象所共有。
对于共有类属性,在类外可通过类对象和实例对象访问。
例如:
class Person:
name = "张三" # 共有类属性
__age = 18 # 私有类属性
p = Person()
print(p.name) # 通过实例对象访问共有属性
print(Person.name) # 通过类对象访问共有属性
"""
输出结果:
张三
张三
"""
私有的类属性在类外不能被访问,否则会报异常。
实例属性
- 在类中通过self.xxx或者实例对象.xxx定义的属性就是实例属性。
- 每个实例属性仅在当前实例对象中有,若想要所有的实例对象有该实例属性,则需要在__init __初始化方法中定义该实例属性。
如下代码所示定义实例属性:
class Person:
def __init__(self):
self.name = "张三" # 定义实例属性
p = Person()
print(p.name)
或者
class Person:
pass
p = Person()
p.name = "张三" # 定义实例属性
print(p.name)
通过类对象修改类属性的值
也许我们会想到通过实例对象去修改类属性的值,那么它是这样的:
class Person:
sex = "男"
def __init__(self):
self.sex = "女"
p = Person()
print(p.sex)
print(Person.sex)
"""
输出结果:
女
男
"""
也可能是这样想的:
class Person:
sex = "男"
p = Person()
p.sex = "女"
print(p.sex)
print(Person.sex)
"""
输出结果:
女
男
"""
本质上它们是一样的,并没有修改类属性的值,只是在当前的实例对象中添加了一个和类属性同名实例属性,并且之后如果通过实例对象去引用该名称的属性,实例属性会强制屏蔽掉类属性,即引用的是实例属性,除非删除了该实例属性。
只有通过类对象才能修改类属性的值
class Person:
sex = "男"
Person.sex = "女"
print(Person.sex)
"""
输出结果:
女
"""
类方法
类对象拥有的方法就是类方法,需要用修饰器@classmethod来标识其为类方法,对于类方法,第一个参数必须是类对象,一般以cls作为第一个参数(当然可以用其他名称的变量作为其第一个参数,但是大部分人都习惯以'cls'作为第一个参数的名字,就最好用'cls'了),能够通过实例对象和类对象去访问。
class People(object):
country = 'china'
#类方法,用classmethod来进行修饰
@classmethod
def getCountry(cls):
return cls.country
p = People()
print(p.getCountry()) #可以用过实例对象引用
print(People.getCountry()) #可以通过类对象引用
"""
输出结果:
china
china
"""
类方法还有一个作用就是可以修改类属性:
class People(object):
country = 'china'
#类方法,用classmethod来进行修饰
@classmethod
def getCountry(cls):
return cls.country
@classmethod
def setCountry(cls,country):
cls.country = country
p = People()
print(p.getCountry()) #可以用过实例对象引用
print(People.getCountry()) #可以通过类对象引用
p.setCountry('English')
print(p.getCountry())
print(People.getCountry())
"""
输出结果:
china
china
English
English
"""
静态方法
如果一个方法传递的参数不是和实例属性有关,那么就可以把它定义成静态方法,需要通过修饰器@staticmethod来进行修饰,静态方法不需要多定义参数。
class People(object):
country = 'china'
@staticmethod
def getCountry():
return People.country
print (People.getCountry())
总结
从类方法和实例方法以及静态方法的定义形式就可以看出来,类方法的第一个参数是类对象cls,那么通过cls引用的必定是类对象的属性和方法;而实例方法的第一个参数是实例对象self,那么通过self引用的可能是类属性、也有可能是实例属性(这个需要具体分析),不过在存在相同名称的类属性和实例属性的情况下,实例属性优先级更高。静态方法中不需要额外定义参数,因此在静态方法中引用类属性的话,必须通过类对象来引用。