在实际工作当中存在一个开放封闭原则

1.对扩展是开放的

    为什么要对扩展开放呢?

    我们说,任何一个程序,不可能在设计之初就已经想好了所有的功能并且未来不做任何更新和修改。所以我们必须允许代码扩展、添加新功能。

  2.对修改是封闭的

    为什么要对修改封闭呢?

    就像我们刚刚提到的,因为我们写的一个函数,很有可能已经交付给其他人使用了,如果这个时候我们对其进行了修改,很有可能影响其他已经在使用该函数的用户。

装饰器完美的遵循了这个开放封闭原则。

首先装饰器存在的意义是在不改变调用原函数的的情况下在原函数的前后增加功能

下面就是一个简单的装饰器

import time  

def func1():
print('in func1') def timer(func):
def inner():
start = time.time()
func()
print(time.time() - start)
return inner func1 = timer(func1) #在这里的赋值调用可以用一个叫做语法糖的东东来替代,那就是@timer
func1() 再来一段代码看看
import time
def timer(func):
def inner():
start = time.time()
func()
print(time.time() - start)
return inner @timer #==> func1 = timer(func1) 一般来说@timer需要紧挨着下面被装饰的函数
def func1(): #@timer要紧挨着我,否则会报错
print('in func1') func1()

到这里,我们可以简单的总结一下:

  装饰器的本质:一个闭包函数

  装饰器的功能:在不修改原函数及其调用方式的情况下对原函数功能进行扩展

还有最后一个问题要解决,刚刚我们讨论的装饰器都是装饰不带参数的函数,现在要装饰一个带参数的函数怎么办呢?

def timer(func):
def inner(a):
start = time.time()
func(a)
print(time.time() - start)
return inner @timer
def func1(a):
print(a) func1(1) 其实装饰带参的函数并不是什么难事,但假如你有两个函数,需要传递的参数不一样呢?
import time
def timer(func):
def inner(*args,**kwargs):
start = time.time()
re = func(*args,**kwargs)
print(time.time() - start)
return re
return inner @timer #==> func1 = timer(func1)
def func1(a,b):
print('in func1') @timer #==> func2 = timer(func2)
def func2(a):
print('in func2 and get a:%s'%(a))
return 'fun2 over' func1('aaaaaa','bbbbbb')
print(func2('aaaaaa')) 现在参数的问题已经完美的解决了,可是如果你的函数是有返回值的呢?
import time
def timer(func):
def inner(*args,**kwargs):
start = time.time()
re = func(*args,**kwargs)
print(time.time() - start)
return re
return inner @timer #==> func2 = timer(func2)
def func2(a):
print('in func2 and get a:%s'%(a))
return 'fun2 over' func2('aaaaaa','bbbbbb')
print(func2('aaaaaa')) 刚刚那个装饰器已经非常完美了,但是正常我们情况下查看函数的一些信息的方法在此处都会失效
def index():
    '''这是一个主页信息'''
print('from index') print(index.__doc__)
#查看函数注释的方法
print(index.__name__) #查看函数名的方法 为了不让他们失效,我们还要在装饰器上加上一点来完善它 from functools import wraps
def deco(func):
@wraps(func) #加在最内层函数正上方
def wrapper(*args,**kwargs):
return func(*args,**kwargs)
return wrapper @deco
def index():
'''哈哈哈哈'''
print('from index') print(index.__doc__)
print(index.__name__) 装饰器的固定格式,这个基本要记住的
def timer(func):
def inner(*args,**kwargs):
'''执行函数之前要做的'''
re = func(*args,**kwargs)
'''执行函数之后要做的'''
return re
return inner 装饰器的固定格式wraps
from functools import wraps

def deco(func):
@wraps(func) #加在最内层函数正上方
def wrapper(*args,**kwargs):
return func(*args,**kwargs)
return wrapper

带参数的装饰器

假如你有成千上万个函数使用了一个装饰器,现在你想把这些装饰器都取消掉,你要怎么做?

一个一个的取消掉? 没日没夜忙活3天。。。

过两天你领导想通了,再让你加上。。。

def outer(flag):
def timer(func):
def inner(*args,**kwargs):
if flag:
print('''执行函数之前要做的''')
re = func(*args,**kwargs)
if flag:
print('''执行函数之后要做的''')
return re
return inner
return timer @outer(False)
def func():
print(111) func()

多个装饰器装饰同一个函数

有些时候,我们也会用到多个装饰器装饰同一个函数的情况。

def wrapper1(func):
def inner():
print('wrapper1 ,before func')
func()
print('wrapper1 ,after func')
return inner def wrapper2(func):
def inner():
print('wrapper2 ,before func')
func()
print('wrapper2 ,after func')
return inner @wrapper2
@wrapper1
def f():
print('in f') f()



最新文章

  1. libvirt 安装篇
  2. Myeclipse下不用dom4j等解析xml文档
  3. Linux Shell编程(12)——操作符
  4. hook研究结果备忘
  5. yii2源码学习笔记
  6. MySQL批量更新死锁案例分析--转载
  7. 重点+超详细:ajax和json及案例
  8. bzoj4826 [Hnoi2017]影魔
  9. 深入理解springAOP切面的特性
  10. Spark环境搭建(二)-----------HDFS shell 常用操作
  11. cadence单一原理图库的设计
  12. 关于TCP窗口大小
  13. Windows Community Toolkit 3.0 - InfiniteCanvas
  14. 【转】python 退出程序的方式
  15. aes 和 Md5 分析
  16. 唤醒实验(java
  17. [java,2017-05-16] java中清空StringBuffer的方法以及耗费时间比较
  18. 【小程序】当遇到bindTap绑定无法跳转到tabbar页面时
  19. Linux中 Lua 访问Sql Server的配置方法
  20. How to manage IntelliJ IDEA projects under Version Control Systems

热门文章

  1. 7.1 Spring原理
  2. 0.spring cloud目录
  3. 了解dubbo+zookeeper
  4. Java中的基本数据类型和引用类型
  5. css三类选择器 用法 引用
  6. CI集成Smarty的实现方式
  7. Java工具类-HttpUtil
  8. java实现一个简单的验证码生成器
  9. 开源的图片查看库之PhotoView
  10. Java 谷歌浏览器开发必备插件