1、首先,有个原来写好的函数,完成一定的功能,比如下面的,就打印一句话(某程序被调用)。简单点,容易帮我们想清楚程序是怎么执行的。

 '''
原函数
'''
def fun1():
print("fun1 is called.") fun1()

结果:

2、然后,有个需求,要在原来的函数上加个功能,我们简单点,就再打印一句话吧(某新功能加入)。

2.1 不懂装饰器,就直白点吧,加一个fun2

 '''
在原来的函数上加个功能
再打印一句话吧(某新功能加入)。
'''
def fun2():
print("new fun is called") '''
原函数
打印一句话(某程序被调用),
'''
def fun1():
fun2()
print("fun1 is called.") fun1()

结果:

这样子,结果是我们想要的,先执行新功能的程序,再执行原功能的程序,但是有个问题,原来的程序中加入了新的代码,示例中的13行。

写程序有个原则,叫封闭,就是原来跑得好好的代码不要再去改它,免得改来改去得再出问题。所以,我们得改。

2.2 不修改源代码,我们把原函数名字作为参数传给新功能的函数,先跑新功能,完了再掉用作为参数传进去的原功能

 '''
在原来的函数上加个功能
再打印一句话吧(某新功能加入)。
'''
def fun2(funname):
print("new fun is called")
funname() '''
原函数
打印一句话(某程序被调用),
'''
def fun1():
print("fun1 is called.") fun2(fun1)

结果:

也是我们要的结果。然后新问题来了,这种方式,可不是原来直接调用fun1(),那么程序中所有用到fun1()的地方都得改成fun2(fun1),如果需要实现对就功能进行加强的地方很多,可以想象,你得找到所有的地方进行修改,程序大了,可不能这么干,万一漏了呢?我们还得改。

2.3 不改源代码,不改调用方式

把原来的程序在新程序中返回,然后再赋值给fun1,这样执行会是什么结果呢?

 '''
在原来的函数上加个功能
再打印一句话吧(某新功能加入)。
'''
def fun2(funname):
print("new fun is called")
return funname '''
原函数
打印一句话(某程序被调用),
'''
def fun1():
print("fun1 is called.") fun1=fun2(fun1) fun1()

结果:

在这个程序里,调用fun1()可以获得和原来一样的结果,但是,这里有个问题,“new fun is called”是在16行的时候,运行fun2时出来的,“fun1 is called”是在18行运行fun1时出来的,也就是说这个程序实际上并没有将fun1和fun2关联起来,它并没有在fun1()调用的时候同时实现新旧两个功能,它和以下代码是一样的

 def fun2():
print("new fun is called")
return
def fun1():
print("fun1 is called.")
fun2()
fun1()

这就没有装饰的意思了
2.4 不改源代码,不改调用方式,一次调用执行实现新旧两个功能

 '''
在原来的函数上加个功能
再打印一句话吧(某新功能加入)。
'''
def fun2(funname):
def inner():
print("new fun is called")
funname()
return inner '''
原函数
打印一句话(某程序被调用),
'''
def fun1():
print("fun1 is called.") fun1=fun2(fun1) fun1()

在这个代码里,fun1作为参数传给fun2,在fun2中定义了一个函数,在这个嵌套定义的函数中实现了新功能,那么fun2干嘛呢?它只是将这个新功能的地址作为返回值返回。我们拿到这个返回值,重新赋值给fun1,那么再调用fun1的时候,实际上是调用的fun2中定义的新功能函数。
结果:

我们可以通过debug运行,给每行打上断点,就能知道最终的结果是在运行到21行的时候一起出来的。

2.5 在以上基础上,可以看到旧功能是在fun1中实现的,新功能是在fun2中实现的,调用的还是fun1,多了一句话,即19行,传参+复制,会不会搞不清楚,于是,简写一下

 '''
在原来的函数上加个功能
再打印一句话吧(某新功能加入)。
'''
def fun2(funname):
def inner():
print("new fun is called")
funname()
return inner '''
原函数
打印一句话(某程序被调用),
'''
@fun2
def fun1():
print("fun1 is called.") fun1()

将@fun2写在fun1之前,就表示fun2写了新功能,它是在fun1的基础上来的,把fun1妆点了一下门面,有点加强了--装饰器这个名字很贴切的。

2.6 函数的定义是为了重复调用,可以少写代码,现在我们有fun2(新功能),可以想象,可能有很多其他的fun**(旧功能)需要妆点,如果原来的fun**本身带参数,那么就会报错。

 '''
在原来的函数上加个功能
再打印一句话吧(某新功能加入)。
'''
def fun2(funname):
def inner():
print("new fun is called")
funname()
return inner '''
原函数
打印一句话(某程序被调用),
'''
@fun2
def fun1():
print("fun1 is called.") @fun2
def fun3(arg1):
print("fun3 welcome: %s "%arg1) fun1()
fun3("susen")

结果:

所以,我们需要用非固定参数来解决这个问题。在装饰器的inner函数用非固定参数*args,**kwargs,这样,不管原来的函数有多少参数,都可以调用了。

 '''
在原来的函数上加个功能
再打印一句话吧(某新功能加入)。
'''
def fun2(funname):
def inner(*args,**kwargs):
print("new fun is called")
funname(*args,**kwargs)
return inner '''
原函数
打印一句话(某程序被调用),
'''
@fun2
def fun1():
print("fun1 is called.") @fun2
def fun3(arg1):
print("fun3 welcome: %s "%arg1) fun1()
fun3("susen")

结果:

3、总结

按照上面一步步得来,应该能明白装饰器是怎么工作的了,基础知识有:非固定参数传参、函数名实际上也是一个变量(内存地址)、嵌套函数。明白以后,之后写代码就套用格式就完了。

最新文章

  1. 注册asp.net 4.0 到iis
  2. vs2013如何在C++中调用Lua(二)
  3. Octopus系列之价格计算公式
  4. org.springframework.web.HttpRequestMethodNotSupportedException: Request method 'PUT' not supported
  5. 2016年12月3日 星期六 --出埃及记 Exodus 20:24
  6. Pre-Query trigger in Oracle D2k / Oracle Forms
  7. fgets和scanf的区别
  8. 【leetcode边做边学】二分查找应用
  9. centos主机建立ssh互信
  10. 三千万数据量下redis2.4的一统计情况
  11. python3.6----datetime.timedelta
  12. 学JAVA第十七天,接口与抽象方法
  13. STS的安装与简单使用
  14. 【386】operator 的 itemgetter、slice、and_、or_
  15. [SQL]查询最新的数据
  16. XML5个转义符
  17. Android 混淆代码有关问题总结
  18. springboot-25-springboot 集成 ActiveMq
  19. 【C#】访问泛型中的List列表数据
  20. Ubuntu16.04编译安装tensorflow,2018最新血泪踩坑之后的全面总结!绝对成功!【转】

热门文章

  1. ubuntu下统计目录及其子目录文件个数
  2. Android -- 自定义view实现keep欢迎页倒计时效果
  3. python学习之元组与文件
  4. logstash结合zabbix报警安装部署
  5. Get json formatted string from web by sending HttpWebRequest and then deserialize it to get needed data
  6. STL中map的用法
  7. 关于变量 Objects...objects 和Object[] objects的区别
  8. Spring Boot之Hello World
  9. oracle字段由中文前缀加数字,数字自动增长的实现
  10. Android Studio发布项目到jcenter,一行代码引入Module