一、@property
python的@property是python的一种装饰器,是用来修饰方法的。
作用:
- 可以使用@property装饰器来创建只读属性,@property装饰器会将方法转换为相同名称的只读属性
- 可以与所定义的属性配合使用,这样可以防止属性被修改。
- 对类的私有属性进行操作
1.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。
class DataSet(object):
@property
def method_with_property(self): ##含有@property
return 15
l = DataSet()
print(l.method_with_property()) # 加了@property后,可以用调用属性的形式来调用方法,后面不需要加()。
如果使用property进行修饰后,又在调用的时候,方法后面添加了 ()
, 那么就会显示错误信息:TypeError: 'int' object is not callable
,也就是说添加 @property
后,这个方法就变成了一个属性,如果后面加入了 ()
,那么就是当作函数来调用,而它却不是 callable(可调用)
的。
class DataSet(object):
def method_without_property(self): ##不含@property
return 15
l = DataSet()
print(l.method_without_property) #没有加@property , 必须使用正常的调用方法的形式,即在后面加()
没有使用property修饰,它是一种方法,如果把括号去掉,不会报错,输出的就会是方法存放的地址。
1.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后,可以用调用属性的形式来调用方法,后面不需要加()。
即,如果这样调用
l = DataSet()
l.images = 5
会报错:AttributeError: can't set attribute
当然,如果用户强行调用
l = DataSet()
l._images = 5
那属性就被改了
私有属性:
1)单下划线 _
开头:只是告诉别人这是私有属性,外部依然可以访问更改
2)双下划线__
开头:外部不可通过 instancename
. propertyname
来访问或者更改,而实际将其转化为了 _classname__propertyname
1.3 对类的私有属性进行操作
一般为了程序安全而封装的原因,python中会有一些私有属性,无法直接访问,那么这个时候如果操作他呢,就需要类给提供相应的 getter
/ setter
方法。
1. 通过自定义get,set方法,调用方法的形式去对私有属性进行操作
class People(object):
def get_name(self):
# 返回前还可以对数据做处理,比如格式化等
return self._name
def set_name(self, value):
if not isinstance(value, str): # 在赋值之前添加判断
raise TypeError('People object.name,name must be a str!') # 抛出异常
self._name = value
# 调用的时候仍然是方便快捷的直接赋值:
p1 = People()
p1.set_name('李四')
print(p1.get_name()) # 李四
2. 使用property标注同名函数的形式。这是最终开发中常用的方法。
class People(object):
def __init__(self, name, age):
self.name = name
self.age = age
@property
def name(self):
# 返回前还可以对数据做处理,比如格式化等
return self._name
@name.setter
def name(self, value):
if not isinstance(value, str): # 在赋值之前添加判断
raise TypeError('People object.name,name must be a str!') # 抛出异常
self._name = value
# 调用的时候仍然是方便快捷的直接赋值:
p1 = People('张三', 22)
p1.name = '李四'
print(p1.name) # 李四
p1.name = 33 # TypeError: People object.name,name must be a str!
@name.setter
(类似于 setter
操作),就可以对属性 name
赋值,而且可以加上对所赋值的属性进行判断。
总结
@property
成为属性函数,可以对属性赋值时做必要的检查,比如在setter
方法里加过滤判断条件。此外相比自定义的get
和set
方法,property
保证了代码的短小精悍,对私有属性的操作代码格式化模块化。@property
只有一个参数self
,且要有返回值,所以一般定义属性的get
方法。- 注意
@property
调用方式,没有括号,而是直接调用属性名即可。
=============================================================
二、@staticmethod
普通的类内函数是需要实例化之后才可以调用的,但是 @staticmethod
静态方法无需实例化也可直接调用
class cal:
def __init__(self,x,y):
self.x = x
self.y = y
@staticmethod #静态方法 类或实例均可调用
def cal_test(a,b,c): #改静态方法函数里不传入self 或 cls
print(a,b,c)
# 实例化
c1 = cal(10,11)
c1.cal_test(1,2,3) #>>> 1 2 3
# 静态方法无需实例化也可直接调用
cal.cal_test(1,2,3) #>>> 1 2 3
@classmethod
参考:正确理解Python中的 @staticmethod@classmethod方法