• python知识:装饰器@property有啥用途?


    一、提要

            python的@property是python的一种装饰器,是用来修饰方法的。python @property 装饰器使一个方法可以像属性一样被使用。除此之外,还有几个使用场景,本文将叙述这些使用技巧。

    二、关于属性的约定

    首先看下属性的分类:

    2.1 类绑定属性

    一般类内属性是指定义成与类同一存储单元的属性,可以类访问。而当类实例化成对象后,类变量将做为拷贝加入对象,对象所访问的属性是一份拷贝。这份拷贝修改后易变。

     验证代码

    1. class Stranger(object):
    2. name = 'class name' #类绑定属性
    3. def __init__(self, gender=None ):
    4. self.gender = gender #对象绑定属性
    5. self.name = 'new name' #对象中属性,与类属性的同名拷贝
    6. stan = Stranger('male')
    7. print(1,stan.gender)
    8. stan.gender = 'famel'
    9. print(2,stan.gender)
    10. print(3 stan.name)
    11. print( 4, Stranger.name)

    结果:

    1 male
    2 famel
    3 new name
    4 class name

    2.2 对象绑定属性

    凡用self定义的属性,都是对象绑定属性,

    • 内部调用时都需要加上self.
    • 外部调用时用 instance_name.property_name进行访问
    1. class Stranger(object):
    2. def __init__(self, gender=None ):
    3. self.gender = gender
    4. stan = Stranger('roma')
    5. print(stan.gender)
    6. stan.gender = 'Paras'
    7. print(stan.gender)

    结果: 

    roma
    Paras 

    注意:事实上实例化后的对象,也可以定义属性,外部也可以调用。 

    2.3 私有属性 

            python的私有属性没有编译限定,知识以单下划线_开头,标记此属性是私有的,但是外部也可以自由访问(私有的程度不够多)。另一种是双下划线__开头的属性,可以转化成类属性访问

    • 单下划线_开头:只是告诉别人这是私有属性,外部依然可以访问更改
    • 双下划线__开头:外部不可通过instancename.propertyname来访问或者更改,实际将其转化为了_classname__propertyname

    三、应用@property装饰器

            python的@property是python的一种装饰器,是用来修饰方法的。python @property 装饰器使一个方法可以像属性一样被使用,而不需要在调用的时候带上() 接下来我们会了解什么时候需要使用它,并且在什么场景下需要用到它以及如何合理的使用它。

    python类中@property装饰器,相配合的方法有:

    • setter() 
    • get() 
    • set()

    相配合。
     

    3.1 将一个属性转成方法

            将一个属性转化为一个方法时,我们最好加上一个@property 装饰器来解决这个问题。在方法定义上面加一个@property 装饰器,可以在不改变原有调用方式的同时,来将一个属性改为一个方法。

    1. class Goods():
    2. def __init__(self,unit_price,weight):
    3. self.unit_price = unit_price
    4. self.weight = weight
    5. @property
    6. def price(self):
    7. return self.unit_price * self.weight
    8. lemons = Goods(7,4)
    9. print(lemons.price)

       28    

     上文中,price是方法,现在将方法转化成属性调用。注意这个技巧,要知道原类定义中无price这个属性,这是一个临时产生的结果。类同于excel表格的“计算项”。

    3.2 私有化某些属性

            对于某些属性,不可直接访问。这里的“直接”就是“无条件”的意思;而条件的访问,就需要装饰器 @property,下例是双装饰器@property和@age.setter配合,对_age进行条件隔离的例子:

    1. class Stranger(object):
    2. def __init__(self, gender=None, age=None, job=None):
    3. self.gender = gender
    4. self._age = age # 这里的成员属性_age需要与成员方法age()区分开
    5. self.jobb = job
    6. # 读取age
    7. @property # 实现一个age相关的getter方法
    8. def age(self):
    9. return self._age
    10. # 设置age
    11. @age.setter # 实现一个age相关的setter方法
    12. def age(self, value):
    13. if isinstance(value, int):
    14. self._age = value
    15. else:
    16. raise ValueError("'int' type need")
    17. if __name__ == "__main__":
    18. # 创建一个“妹子”
    19. meizi = Stranger()
    20. meizi.age = 18 # 使用时注意是.age,不是._age
    21. print("年龄:{age}".format(age=meizi.age))

    注意事项:

    • 属性名与方法名一定要区分开,不然会进入死循环(self._age,def age())
    • 实例化的对象使用属性时,不是调用属性(meizi._age),而是用的方法名(meizi.age)
    • @property其实就是实现了getter功能; @xxx.setter实现的是setter功能;还有一个 @xxx.deleter实现删除功能
    • 定义方法的时候 @property必须在 @xxx.setter之前,且二者修饰的方法名相同(age())
    • 如果只实现了 @property(而没有实现@xxx.setter),那么该属性为 只读属性

    3.3 关联性修改

    比如,我们输入了first_name、last_name可以得出fullname,下面代码可以实现全名的属性获取。而反过来,对全名进行修改后,如何将连带的first_name、last_name同步进行修改?。下文中的 @fullname.setter就是解决此类问题的。

    1. class Person():
    2. def __init__(self, first_name, last_name):
    3. self.first = first_name
    4. self.last = last_name
    5. @property
    6. def fullname(self):
    7. return self.first + ' ' + self.last
    8. @fullname.setter
    9. def fullname(self, name):
    10. first_name, last_name = name.split()
    11. self.first = first_name
    12. self.last = last_name
    13. def email(self):
    14. return '{}.{}@email.com'.format(self.first, self.last)
    15. person = Person('zhang', 'san')
    16. print(person.fullname)
    17. print(person.last)
    18. print(person.first)
    19. person.fullname = 'li si'
    20. print(person.fullname)
    21. print(person.last)
    22. print(person.first)

    3.4 删除属性的deleter方法

            和setter 方法类似,当我们需要删除一个属性时,我们会使用deleter 方法。

            你可以像定义setter 方法一样来定义一个setter 方法,使用相同的方法名,并在方法上添加@{methodname}.deleter 装饰器 。

    1. class Person():
    2. def __init__(self, first_name, last_name):
    3. self.first = first_name
    4. self.last = last_name
    5. @property
    6. def fullname(self):
    7. return self.first + ' ' + self.last
    8. @fullname.setter
    9. def fullname(self, name):
    10. first_name, last_name = name.split()
    11. self.first = first_name
    12. self.last = last_name
    13. @fullname.deleter
    14. def fullname(self):
    15. self.first = None
    16. self.last = None
    17. def email(self):
    18. return '{}.{}@email.com'.format(self.first, self.last)
    19. person = Person('zhang', 'san')
    20. print(person.fullname)
    21. print(person.last)
    22. print(person.first)
    23. del person.fullname
    24. print(person.last)
    25. print(person.first)

     

    四、property()函数原理

    使用该函数可以将方法直接变成属性,与@property类同。

    • 函数接口:property(fget=None, fset=None, fdel=None, doc=None)

    使用property的代码示例:

    1. class Stranger(object):
    2.     def __init__(self, gender=None, age=None, job=None):
    3.         self.gender = gender
    4.         self._age = age
    5.         self.jobb = job
    6.     # 设置_age
    7.     def set_age(self, age):
    8.         if isinstance(age, int):
    9.             self._age = age
    10.         else:
    11.             raise ValueError("'int' type need")
    12.     # 读取_age
    13.     def get_age(self):
    14.         return self._age
    15.     
    16.     # 使得实例化对象可以利用.age方式来访问
    17.     age = property(get_age, set_age)
    18. if __name__ == "__main__":
    19.     # 创建一个“妹子”
    20.     meizi = Stranger()
    21.     meizi.age = 18
    22.     print("年龄:{age}".format(age=meizi.age))

    # 输出:
    #年龄:18

  • 相关阅读:
    通过uboot命令开启核隔离
    含泪推荐5款WIN10装机必备的软件
    使用 Azure 部署静态网页
    Spring Boot常规知识整理
    机器学习笔记 - FlowNet:使用卷积网络学习光流
    idea配置tomcat的方法(详细图文步骤)
    Centos7 设置定时任务
    省重点实验室成果转化:24h在线的专业数字朋友,不用欠人情的网上解答神器~
    数据库:Hive转Presto(五)
    桶排序以及排序内容大总结
  • 原文地址:https://blog.csdn.net/gongdiwudu/article/details/125402452