zoukankan      html  css  js  c++  java
  • Python装饰器之 property()

    1. 何为装饰器?

      官方定义:装饰器是一个很著名的设计模式,经常被用于有切面需求的场景,较为经典的有插入日志、性能测试、事务处理等。装饰器是解决这类问题的绝佳设计,有了装饰器,我们就可以抽离出大量函数中与函数功能本身无关的雷同代码并继续重用。概括的讲,装饰器的作用就是为已经存在的对象添加额外的功能。

       Python中总共包括三个内置装饰器:

        ① staticmethod

        ② classmethod

        ③ property

      更加详细的解释,请点击(传送门

    2. 属性函数 property() 浅谈

      2.1 为什么要使用 property?

          通常,我们在访问属性和给属性赋值的时候,都是对 类和实例 __dict__ 打交道的;但如果我们想要规范属性访问,有两种方式可用:①数据描述符 ,②. property() 属性函数。

          然而,我们知道,描述符相对比较复杂,对于新手来说,用起来很吃力,那么不妨试试property(),相对于描述符这个大的进程,property就相当于线程。

      2.2 函数原型:

        property(fget=None, fset=None, fdel=None, doc=None)

      2.3 普通方法定义:

        假设 calss Normal中有一个私有变量 __x,如下代码所示:

    复制代码
     1 #code 1
     2 
     3 class Normal:
     4     def __init__(self):
     5         self.__x = None
     6     def getx(self):
     7         return self.__x
     8     def setx(self, value):
     9         self.__x = value
    10     def delx(self):
    11         del self.__x
    12 
    13 
    14 tN = Normal()
    15 print(tN.__count)
    16 
    17 #输出结果(报错了)
    18 Traceback (most recent call last):
    19   File "C:/Users/Administrator/AppData/Local/Programs/Python/Python35/property.py", line 15, in <module>
    20     print(tN.__count)
    21 AttributeError: 'Normal' object has no attribute '__count'
    复制代码

      为啥报错了呢?因为 实例tN的属性 __x 为私有属性,不能直接访问,为此我们只能调用内部定义的 方法;

    复制代码
    1 tN = Normal()
    2 tN.setx(10)
    3 print(tN.getx())
    4 
    5 #输出结果:
    6 10
    复制代码

      嗯,使用内部的方法,可以容易的得到实例的或者类的私有属性值;

      然而,如果我那一天兴致来潮,把 class Normal 的 setx方法名改成了其它(如 Normal_setx),外部很多地方用到了该函数,是不是我需要一个一个的去找该方法的调用地点,然后一个一个的改呢?

      c语言或许会,但Python,一个智能化的语言,怎么会这么点事都解决不了呢?

      那么,该如何解决以上请呢?

      哈哈,其实有两种方法哦,听我慢慢道来哦!

      方法一:采用 属性函数property()

    复制代码
     1 #改进方法一
     2 
     3 class Normal:
     4     def __init__(self):
     5         self.__x = None
     6     def getx(self):
     7         print('getx(): self.__x=', self.__x)
     8         return self.__x
     9     def setx(self, value):
    10         self.__x = value
    11         print('setx()')
    12     def delx(self):
    13         print('delx()')
    14         del self.__x
    15 
    16     y = property(getx, setx, delx, "I'm a property")
    17 
    18 
    19 tN=Normal()
    20 tN.y=10
    21 tN.y
    22 del tN.y
    23 
    24 #输出结果:
    25 setx()
    26 getx(): self.__x= 10
    27 delx()
    复制代码

        哈哈,直接把方法当属性来操作了,好流弊的样子!

      方法二:采用 @property 装饰器

    复制代码
     1 #改进方法二
     2 
     3 class Normal:
     4     
     5     def __init__(self):
     6         self.__x = None
     7 
     8     @property
     9     def xx(self):
    10         print('getx(): self.__x=', self.__x)
    11         return self.__x
    12     
    13     @xx.setter
    14     def xx(self, value):
    15         self.__x = value
    16         print('setx()')
    17 
    18     @xx.deleter
    19     def xx(self):
    20         print('delx()')
    21         del self.__x
    22 
    23 
    24 tN=Normal()
    25 tN.xx=10
    26 tN.xx
    27 del tN.xx
    28 
    29 #输出结果信息:
    30 setx()
    31 getx(): self.__x= 10
    32 delx()
    复制代码

        哈哈,怎么样,跟方法一输出一样的结果,证明,这两种方法都可行的(注意哦,第一个一定是 @property(替代getter哦,不然会报错))。

  • 相关阅读:
    Android TP出现小圆点解决方法
    Android的SAFE MODE(安全模式)
    Linux TCP透传到OneNET
    C读取BMP数据
    OLED显示BMP数据
    电脑出现DNS错误无法上网怎么办
    办公室如果没有网络怎么办呢?
    微信扫一扫获取地理位置
    win10系统程序与功能查找,卸载程序
    手把手学习数据库
  • 原文地址:https://www.cnblogs.com/leesen934/p/9674683.html
Copyright © 2011-2022 走看看