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 classNormal:4 def __init__(self):5 self.__x =None6 defgetx(self):7 return self.__x
8 defsetx(self, value):9 self.__x =value10 defdelx(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
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 classNormal:4 def __init__(self):5 self.__x =None6 defgetx(self):7 print('getx(): self.__x=', self.__x)8 return self.__x
9 defsetx(self, value):10 self.__x =value11 print('setx()')12 defdelx(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.y22 deltN.y23
24 #输出结果:
25 setx()26 getx(): self.__x= 10
27 delx()
哈哈,直接把方法当属性来操作了,好流弊的样子!
方法二:采用 @property 装饰器
1 #改进方法二
2
3 classNormal:4
5 def __init__(self):6 self.__x =None7
8 @property9 defxx(self):10 print('getx(): self.__x=', self.__x)11 return self.__x
12
13 @xx.setter14 defxx(self, value):15 self.__x =value16 print('setx()')17
18 @xx.deleter19 defxx(self):20 print('delx()')21 del self.__x
22
23
24 tN=Normal()25 tN.xx=10
26 tN.xx27 deltN.xx28
29 #输出结果信息:
30 setx()31 getx(): self.__x= 10
32 delx()
哈哈,怎么样,跟方法一输出一样的结果,证明,这两种方法都可行的(注意哦,第一个一定是 @property(替代getter哦,不然会报错))。