类方法、类属性
类方法需要用 装饰器 @classmethod 来标识
• 类方法的 第一个参数 应该是 cls
• 由 哪一个类 调用的方法,方法内的 cls 就是 哪一个类的引 用 • 这个参数和 实例方法 的第一个参数是 self 类似 • 通过 类名. 调用 类方法,调用方法时,不需要传递 cls 参数 • 在方法内部
• 可以通过 cls. 访问类的属性
• 也可以通过 cls. 调用其他的类方法
class Cat:
count=0 #类属性 所有对象共享
def __init__(self, name): #初始化对象属性
self.name = name
Cat.count+=1
def eat(self): #对象方法
print(f"{self.name} 爱吃鱼")
#类方法 所有对象共享
@classmethod
def 类方法名(cls):
pass
#默认情况下使用 print() 函数打印对象变量时, 会输出对象的内存地址信息,
# 如果想要自定义打印对象变量时的信息, 则可以使用 __str__ 方法
def __str__(self):
return f"我是小猫:{self.name}"
if __name__ == '__main__':
tom = Cat("Tom")
print(tom) #我是小猫: Tom
实例属性、实例方法
带self的都是实例属性或方法
class Bicycle:
wheel=2 #类属性
def __init__(self,color,speed,wheel):
self.color = color #实例属性
self.speed = speed #实例属性
def message(self): #实例方法
print(f"这辆{self.color}的单车有{self.wheel}个轮子")
私有属性、私有方法
私有属性: __属性名
私有方法: __方法名()
说明: 当属性和方法只需要在类定义内部使用时, 就可以使用私有属性和私有方法
特点: 在类定义外部, 无论是通过类对象还是实例对象均无法获取私有属性和调用私有方法
静态方法
在开发时,如果需要在 类 中封装一个方法,这个方法:
这个时候,可以把这个方法封装成一个 静态方法
class Demo:
@staticmethod
def 静态方法名(): #通过类名.调用静态方法
pass
@Property
作用
@property装饰器来创建只读属性,@property装饰器会将方法转换为相同名称的只读属性,可以与所定义的属性配合使用,这样可以防止属性被修改。
使用场景
1.修饰方法,是方法可以像属性一样访问。
class DataSet(object):
@property
def method_with_property(self): ##含有@property
return 15
def method_without_property(self): ##不含@property
return 15
l = DataSet()
print(l.method_with_property) # 加了@property后,可以用调用属性的形式来调用方法,后面不需要加()。
print(l.method_without_property()) #没有加@property , 必须使用正常的调用方法的形式,即在后面加()
两个都输出为15。
如果使用property进行修饰后,又在调用的时候,方法后面添加了(), 那么就会显示错误信息:TypeError: ‘int’ object is not callable,也就是说添加@property 后,这个方法就变成了一个属性,如果后面加入了(),那么就是当作函数来调用,而它却不是callable(可调用)的。
没有使用property修饰,它是一种方法,如果把括号去掉,不会报错输出的就会是方法存放的地址。
2.与所定义的属性配合使用,这样可以防止属性被修改。
由于python进行属性的定义时,没办法设置私有属性,因此要通过@property的方法来进行设置。这样可以隐藏属性名,让用户进行使用的时候无法随意修改。
class DataSet(object):
def __init__(self):
self._images = 1
self._labels = 2 #定义属性的名称
@property
def images(self): #方法加入@property后,这个方法相当于一个属性,这个属性可以让用户进行使用,而且用户有没办法随意修改。
return self._images
@property
def labels(self):
return self._labels
l = DataSet()
#用户进行属性调用的时候,直接调用images即可,而不用知道属性名_images,因此用户无法更改属性,从而保护了类的属性。
print(l.images) # 加了@property后,可以用调用属性的形式来调用方法,后面不需要加()。