Python装饰器详解

装饰器简介:

  装饰器本质上是一个Python函数,它可以让其他函数在不需要做任何代码变动的前提下增加额外功能,装饰器的返回值也是一个函数对象。它经常用于有切面需求的场景,比如:插入日志、性能测试、事务处理、缓存、权限校验等场景。装饰器是解决这类问题的绝佳设计,有了装饰器,我们就可以抽离出大量与函数功能本身无关的雷同代码并继续重用。

一个简单装饰器写法:

  在写装饰器之前我们需要简单了解下python语法糖@的用法 http://www.cnblogs.com/Egbertbaron/p/7242515.html

def debug(func):
    def wrapper():
        print("[DEBUG]: enter {}()".format(func.__name__))  # 输出原函数名字
        return func()
    return wrapper()  # 回调包装过的函数

@debug
def say_hello():
    print("Hello world!")

运行结果:
[DEBUG]: enter say_hello()
Hello world!

  以上代码就实现了一个简单的装饰器。他对原函数进行了包装并返回了另一个函数,额外添加了一个输出原函数名字的功能,但是这样做有一个问题,那就是原函数 say_hello() 不能就行传参,并且返回的函数 wrapper() 也并不能接受参数,所以我们这个代码还得进一步的完善。例如:

def debug(func):
    def wrapper(something):  # 指定和原函数一模一样的参数
        print("[DEBUG]: enter {}()".format(func.__name__))
        return func(something)
    return wrapper  # 回调包装过的函数

@debug
def say_hello(something):
    print("Hello world!{}".format(something))

say_hello(‘Mr.hou‘)  # 给原函数传一个参数

运行结果:
[DEBUG]: enter say_hello()
Hello world!Mr.hou

  刚解决了传参数问题,得!新问题又来了!那么多的函数每个函数又都有自己函数,包装函数有可能把没函数的参数添加进去,所以我们在这里有可以用到python提供的 可变参数*args 和 关键字参数**kwargs (这里有关于 *args 和 **kwargs 的介绍http://www.cnblogs.com/Egbertbaron/p/7242515.html),有了这俩参数,装饰器就可以用于带任意参数目标函数了。修改后的代码如下:

def debug(func):
    def wrapper(*args, **kwargs):  # 把指定参数变为可变参数和关键字参数
        print("[DEBUG]: enter {}()".format(func.__name__))
        return func(*args, **kwargs)
    return wrapper  # 回调包装过的函数

@debug
def say_hello(something):
    print("Hello world!{}".format(something))
时间: 2024-08-06 23:24:01

Python装饰器详解的相关文章

Python装饰器详解,详细介绍它的应用场景

装饰器的应用场景 附加功能 数据的清理或添加: 函数参数类型验证 @require_ints 类似请求前拦截 数据格式转换 将函数返回字典改为 JSON/YAML 类似响应后篡改 为函数提供额外的数据 mock.patch 函数注册 在任务中心注册一个任务 注册一个带信号处理器的函数 不同应用场景下装饰器实现 函数注册表 简单注册表 funcs = [] def register(func): funcs.append(func) return func @register def a(): r

这是我见过最全面的Python装饰器详解!没有学不会这种说法!

上面是按下按钮1就存款,否则则取款. 不对,存取款要输入密码啊!!!所以,我们要加密码验证代码. 进群:125240963  即可获取数十套PDF哦! 可以看到,虽然实现了密码验证功能,但是代码冗余度比较高,而且现在只模拟了取款和存款功能,然而还有查询功能,转账功能等等,那么冗余度就更高了,而且相对于取款和存款函数来说,复用性没有那么高,所以我们要进一步优化代码,把验证函数写到取款和存款函数内部. 有没有什么方法,可以在不改变原函数以及原函数的调用的情况下扩展原函数的功能呢?当然是有的,这就是p

python装饰器详解,多层装饰器,及带参数的装饰器。

pyhon学习有一段时间了,今天又碰到了Django的中间件,其实Django里面的中间件,就是用了多层的装饰器,然后去了解了一下多层装饰器,感觉有写东西差不多快忘了,也可能前面没学好. 现在重新记录,学习下. 普通装饰器: def warp(func): print('我是装饰器,碰到需要装饰的函数,一开始执行这里') def inner(*args, **kwargs): print('这里才是真正的装饰开始!') res = func(*args, **kwargs) print('装饰结

Python全栈开发之8、装饰器详解

一文让你彻底明白Python装饰器原理,从此面试工作再也不怕了. 一.装饰器 装饰器可以使函数执行前和执行后分别执行其他的附加功能,这种在代码运行期间动态增加功能的方式,称之为“装饰器”(Decorator),装饰器的功能非常强大,但是理解起来有些困难,因此我尽量用最简单的例子一步步的说明这个原理. 1.不带参数的装饰器 假设我定义了一个函数f,想要在不改变原来函数定义的情况下,在函数运行前打印出start,函数运行后打印出end,要实现这样一个功能该怎么实现?看下面如何用一个简单的装饰器来实现

Python 3 之 装饰器详解

------------ 装饰器 ----------------------------------------------------- 什么是装饰器 装饰器是为函数和类指定管理代码的一种方式.装饰器本身的形式是处理其他的可调用对象的可调用对象(如函数).正如我们在本书前面所见到过的,Python装饰器以两种相关形式呈现: 函数装饰器在函数定义的时候进行名称重绑定,提供一个逻辑层来管理函数和方法或随后对它们调用. 类装饰器在类定义的时候进行名称重绑定,提供给一个逻辑层来管理类,或管理随后调用

python之装饰器详解

这几天翻看python语法,看到装饰器这里着实卡了一阵,最初认为也就是个函数指针的用法,但仔细研究后发现,不止这么简单. 首先很多资料将装饰器定义为AOP的范畴,也就是Aspect Oriented Programming面向切面编程的概念,不懂AOP不要紧,只要有函数指针的概念,又有嵌套函数的基础知识,看懂此文一点压力都没有. 先说说为什么要有装饰器这么个东西存在吧,这是一种设计模式,较为经典的有插入日志.性能测试.事务处理等等.概括的讲,装饰器的作用就是为已经存在的对象添加额外的功能. 关于

python函数装饰器详解

基础:函数装饰器的表现方式 假如你已经定义了一个函数funcA(),在准备定义函数funcB()的时候,如果写成下面的格式: @funcA def funcB():... 表示用函数funcA()装饰函数funcB().当然,也可以认为是funcA包装函数funcB.它等价于: def funcB():... funcB = funcA(funcB) 也就是说,将函数funcB作为函数funcA的参数,funcA会重新返回另一个可调用的对象(比如函数)并赋值给funcB. 所以,funcA要想作

python基础 之 装饰器详解

装饰器的语法以@开头,接着是装饰器函数的名字.可选参数. 紧跟装饰器声明的是被装饰的函数和被装饰的函数的可选参数,如下: @decorator(dec_opt_args) def func(func_args):   .... 其实总体说起来,装饰器其实也就是一个函数,一个用来包装函数的函数,装饰器在函数声明完成的时候被调用,调用之后声明的函数被换成一个被装饰器装饰过后的函数. 如: def deco(func):    ...    return func @deco def foo():  

16.python全栈之路:装饰器详解

装饰器 一.装饰器的意义 比如:以下函数是供我们调用的,但是我们需要在每个函数中都添加同一个功能,该如何做到呢? 方法一:在每个函数中都加上新加的功能代码块 def f1(): print("新功能") print("F1") def f2(): print("新功能") print("F2") def f3(): print("新功能") print("F3") def f4(): p