参考链接:Python 函数装饰器

  我认为python中的装饰器是一个很厉害的功能,他能瞬间提升代码的逼格,但对于我这样的小白来说,别说为所欲为的使用了,就连简单的尝试一下,却也是难于登天。经过长达半年的努力,我终于找到了大部分装饰器的介绍信息,鲁迅曾经说过,良好的开始就代表了成功的一半,在我看来,鲁迅分明还是太保守,良好的开端无疑代表你已经成功了。比如接下来我们只需要Ctrl+C+V,就可以完成装饰器的学习了,亲爱的小朋友们,你们学会了吗?

给大人看的分割线


装饰器是用来给已经定义好的函数增加功能用的,能在方法运行期间动态增加方法的功能

1)不会改变原来的代码结构。

为什么这么说呢,因为相比于另一种方法:只将函数作为变量传入另一个函数从而来减少重合,使用装饰器并不需要将函数调用改写

#定义这个函数来为类似foo()的需要去增加loging.warn()的全部函数增添这个loging.warn()的功能
#假如有许多函数和foo()一样,都需要增加loging.warn()这行代码,我们就可以定义一个函数,然后将每个需要增加功能的函数作为参数传入
#但是这样必须在用use_logging(foo)去替换原有的foo()这就改变了函数的调用,破坏了原有的代码结构
def use_logging(func):
logging.warn("%s is running" % func.__name__)
func() def foo():
print('i am foo') use_logging(foo)

2)装饰器在不修改函数代码的前提下影响代码的功能;调用python中的函数可以作为变量去传递,基于这一特性,我们可以构建出自己的装饰器;

先来了解一些基础知识:

  1)我们可以将函数作为变量传递。要注意括号,加了括号就会运行函数。

  2)在函数的内部还可以嵌套函数

def hi(name="yasoob"):
print("现在在hi()里面运行") def greet():
return "现在在greet()里面运行" def welcome():
return "现在在welcom()里面运行"
print(greet())
print(welcome())
print("现在你返回到了hi()里面") hi()
# 上面展示了无论何时你调用hi(), greet()和welcome()将会同时被调用。
# 然后greet()和welcome()函数在hi()函数之外是不能访问的,比如

  结果:

现在在hi()里面运行
现在在greet()里面运行
现在在welcom()里面运行
现在你返回到了hi()里面

  

 3)从函数中返回函数

def hi(name="yasoob"):
def greet():
return "now you are in the greet() function" def welcome():
return "now you are in the welcome() function" if name == "yasoob":
return greet
else:
return welcome a = hi()
print(a)
#outputs: <function greet at 0x7f2143c01500> #上面清晰地展示了`a`现在指向到hi()函数中的greet()函数
#现在试试这个 print(a())
#outputs: now you are in the greet() function

  注意下面的 hi()返回的值是greet,加上后面的括号时,就成为了greet()随即开始运行

 hi()() #这会输出 now you are in the greet() function。

  

 4)将函数作为参数传递给另外一个函数

def hi():
return "hi yasoob!" def doSomethingBeforeHi(func):
print("I am doing some boring work before executing hi()")
print(func()) doSomethingBeforeHi(hi) #注意不要加括号
#outputs:I am doing some boring work before executing hi()
# hi yasoob!

  

第一个修饰器

def a_new_decorator(a_func):
def wrapTheFunction():
print('我在a_func执行前做一些事情')
a_func()
print('我在a_func执行后做一些事情')
return wrapTheFunction #记得把这个内定义的函数返回 def a_function_requiring_decoration(): #定义一个函数,等会用修饰器来修饰
print('我是需要用修饰器修饰的函数') #使用修饰器
a_function_requiring_decoration=a_new_decorator(a_function_requiring_decoration)
a_function_requiring_decoration()

  结果:

我在a_func执行前做一些事情
我是需要用修饰器修饰的函数
我在a_func执行后做一些事情

 

真正的装饰器:decorator

  另一种生成被装饰函数的方法:用@

补充----------------

  装饰器,只能在函数运行前做一些事情,也能在函数运行后执行,不过要是有返回结果的就不能在其后执行了,因为有return语句但是可以将装饰器写成这样

def log(parameter1):
def decorator(func):
@functools.wraps(func)
def wrapper(*args,**kw):
s=func(*args,**kw)
print('函数%s()运行消耗时间%.11f'%(func.__name__,parameter1))
return s#前面执行,这里返回
return wrapper
return decorator

  

补充结束------------------

先了解一下@语句

最简单的装饰器

  如果被装饰的函数运行时不需要参数,我们可以这样定义装饰器

def metric(fn):
print('%s executed in %s ms' % (fn.__name__, 10.24))
return fn
@metric
def customize(a,b):
  print("我是被装饰的函数,运行结果%d"%a+b)
customize(1,3) #输出
customize executed in 10.24 ms
我是被装饰的函数,运行结果4

  把@metric放到customize()函数的定义处,相当于执行了语句:

customize=metric(customize)

  由于metric()是一个decorator,返回一个函数,所以,原来的customize()函数仍然存在,只是现在同名的customize变量指向了新的函数,于是调用customize()将执行新函数,即在metric()函数中返回的wrapper中返回的customize函数。

普遍使用的装饰器的格式

  上面的只是最简单的,我们通常使用装饰器的时候都需要被装饰的函数的运行结果,当被装饰的函数需要参数时,使用上面的定义就不行了:

  

decorator实际上也是一个返回函数的函数,不过他返回的函数并不是我们要装饰的函数。
装饰器是在方法执行的过程中动态的为方法增加功能的一种方式,不知道我们能不能想到,这种方式可以通过先运行要增加的功能,然后再运行要修饰的对象函数
而两部过程应该放到一个方法体中,而装饰器返回的就是包含这两步过程的一个函数。
另外需要注意的是参数,装饰器接受的参数是要修饰的方法名,其内部的函数接受的是要修饰的方法需要的参数(用*args,**kw)来表示就好
在这个函数中,先运行计划为函数增加功能的代码,然后运行函数
最后,通过python的@语法将这个装饰器“绑定”到要修饰的方法上
def a_new_decorator(a_func):
def wrapTheFunction():
print('我在a_func执行前做一些事情')
a_func()
print('我在a_func执行后做一些事情')
return wrapTheFunction #记得把这个内定义的函数返回 #不同之处在这里
@a_new_decorator
def a_function_requiring_decoration(): #定义一个函数,等会用修饰器来修饰
print('我是需要用修饰器修饰的函数') a_function_requiring_decoration()

  把@log放到now()函数的定义处,相当于执行了语句:

now = log(now)

  

由于log()是一个decorator,返回一个函数,所以,原来的now()函数仍然存在,只是现在同名的now变量指向了新的函数,于是调用now()将执行新函数,即在log()函数中返回的wrapper()函数。

wrapper()函数的参数定义是(*args, **kw),因此,wrapper()函数可以接受任意参数的调用。在wrapper()函数内,首先打印日志,再紧接着调用原始函数。

解决被修饰的函数会被重命名问题的wraps函数

  使用上面会出现一个问题,获取_name_时会返回被修饰器内置函数的名字

print(a_function_requiring_decoration.__name__)
# Output: wrapTheFunction  应该返回a_function_requiring_decoration
之所以返回的是修饰器里内置的名字,是因为它会重写函数的名字和注释文档

而python提供了 functools.wraps() 来解决这个办法,以如下代码为例说明 functolls.wraps 使用方法

from functools import wraps

def a_new_decorator(a_func):
@wraps(a_func)
def wrapTheFunction():
---snip---
    return a_func
return wrapTheFunction @a_new_decorator
def a_function_requiring_decoration():
---snip-- print(a_function_requiring_decoration.__name__)
# Output: a_function_requiring_decoration

@wraps 接受一个被修饰函数的名字,并加入了复制函数名称、注释文档、参数列表等功能,这让我们能在修饰器里访问函数被修饰之前的属性

怎样为修饰器传入参数:

  

假如装饰器也需要一些参数怎么办呢,比如我们想要输出日志的同时还要输出程序的运行时间.这需要三个函数来完成,绑定最外面的函数,其他的可以看作不变

  假如 a_function_requiring_decoration() 需要参数

def a_function_requiring_decoration(name):
--snip--

  我们需要在定义 wrapTheFunction() 时指定参数即:

def wrapTheFunction(name):
---snip---
    return a_func(name)

  当装饰器不知道要装饰的函数有多少个参数时,可以用 *args 来代替

def wrapTheFunction(*args):
---snip---
    return a_func(*args)

  当需要关键字参数时

def a_function_requiring_decoration(name, age=None, height=None):
  print("I am %s, age %s, height %s" % (name, age, height))

  可以这样

def wrapTheFunction(*args, **kwargs):
  # args是一个数组,kwargs一个字典
  logging.warn("%s is running" % func.__name__)
  return func(*args, **kwargs)

  和两层嵌套的decorator相比,3层嵌套的效果是这样的:

def log(text):
def decorator(func):
def wrapper(*args, **kw):
print('%s %s():' % (text, func.__name__))
return func(*args, **kw)
return wrapper
return decorator

  用法

@log('execute')
def now():
print('2015-3-25')

  和两层嵌套的decorator相比,3层嵌套的效果是这样的:

>>> now = log('execute')(now)

  我们来剖析上面的语句,首先执行log('execute'),返回的是decorator函数,再调用返回的函数,参数是now函数,返回值最终是wrapper函数。

原文链接 >>>点击<<<

python 进阶:修饰器的介绍的更多相关文章

  1. python中用修饰器进行异常日志记录

    当脚本中需要进行的的相同的异常操作很多的时候,可以用修饰器来简化代码.比如我需要记录抛出的异常: 在log_exception.py文件中, import functools import loggi ...

  2. python函数修饰器(decorator)

    python语言本身具有丰富的功能和表达语法,其中修饰器是一个非常有用的功能.在设计模式中,decorator能够在无需直接使用子类的方式来动态地修正一个函数,类或者类的方法的功能.当你希望在不修改函 ...

  3. python 类修饰器

    1. 修改类函数. 场景: 如果要给一个类的所有方法加上计时,并打印出来.demo如下: # -*- coding:utf-8 -*- import time def time_it(fn): &qu ...

  4. python decorator 修饰器

    decorator 就是给函数加一层皮,好用! 1 from time import ctime 2 3 def deco(func): 4 def wrappedFunc(*args, **kwar ...

  5. Python 函数修饰器

    # 一.用函数修饰函数 #!/usr/bin/python3 def decorate_func(func): def call(*args, **kwargs): print('you have c ...

  6. Python多版本管理器-pyenv 介绍及部署记录

    一. pyenv简单介绍 在日常运维中, 经常遇到这样的情况: 系统自带的Python是2.x,而业务部署需要Python 3.x 环境, 此时需要在系统中安装多个Python版本,但又不能影响系统自 ...

  7. Python学习-修饰器 - itemgetter的妙用

    下面这篇对装饰器讲的很好,懂了. http://python.jobbole.com/85056/ <简单 12 步理解 Python 装饰器> 使用装饰器非常简单(见步骤10),但是写装 ...

  8. python 之修饰器

    from functools import update_wrapper def debug(func): def wrapper(): print "[DEBUG]: enter {}() ...

  9. python 通用 修饰器

    import functools def log(option): def dec(func): def swapper(*arg, **karg): functools.update_wrapper ...

随机推荐

  1. 《Unix环境高级编程》读书笔记 第4章-文件和目录

    1. stat结构的基本形式: on error 24. 设备特殊文件 每个文件系统所在的存储设备都由其主.次设备号表示. 设备号所用的数据类型是基本系统数据类型dev_t. 主设备号标识设备驱动程序 ...

  2. FCC高级编程篇之Exact Change

    Exact Change Design a cash register drawer function checkCashRegister() that accepts purchase price ...

  3. js的调试和优化

    一.常见的错误和异常 1.拼写错误 拼写错误,可以有代码的高亮来发现. 2.访问不存在的变量 3.括号不匹配 养成规范的编写习惯,适当应用Tab.空行等. 4.字符串和变量链接错误 采用多加括号来进行 ...

  4. CodeForces-766D Mahmoud and a Dictionary 并查集 维护同类不同类元素集合

    题目链接:https://cn.vjudge.net/problem/CodeForces-766D 题意 写词典,有些词是同义词,有些是反义词,还有没关系的词 首先输入两个词,需要判断是同义还是是反 ...

  5. P1064 金明的预算方案 (依赖性背包问题)

    这道题可以用分组背包来做. 但是分组有两种方式 一种是把主件,主件+附件1,主件+附件2分成一组 组内只能选一个物品 一种是建一颗树,用树形dp的方式去做 第二种更通用,就算物品的依赖关系是森林都可以 ...

  6. 紫书 习题 11-4 UVa 1660 (网络流拆点法)

    这道题改了两天-- 因为这道题和节点有关, 所以就用拆点法解决节点的容量问题. 节点拆成两个点, 连一条弧容量为1, 表示只能经过一次. 然后图中的弧容量无限. 然后求最小割, 即最大流, 即为答案. ...

  7. dll签名两种方法

    以下两种签名方法,都是对csp.dll签名,都不是CA颁发的,且效果不同, 一:通过自建证书签名 下载windows sdk,成功安装后,包括makecert.exe, cert2spc.exe, p ...

  8. 理解Swift中map 和 flatMap对集合的作用

    map和flatMap是函数式编程中常见的概念,python等语言中都有.借助于 map和flapMap 函数可以非常轻易地将数组转换成另外一个新数组. map函数可以被数组调用,它接受一个闭包作为參 ...

  9. OC中的@的作用研究

    OC中的@字符用的频率很的高,其主要作用是为了差别于其它语言的keyword和语法 以下我们来研究一下其应用 1.声明类,协议,延展,权限,属性等 @interface声明类 @protocol声明协 ...

  10. 怎样实如今Windows下编写的代码,直接在Linux下编译

    方法一: 怎样实如今Windows7下编写Linux程序.写完程序以后.不用复制文件,直接在Linux(RHEL6.5)机器上编译最新的代码. 1.首先将Windows的代码目录设置为共享目录: 2. ...