一、为什么要有有参装饰器?

来看之前的无参装饰器

# 无参装饰器
def outter(func):
def wrapper(*args,**kwargs):
start = time.time()
res = func(*args,**kwargs) # 我们需要一个变量接受函数的返回值
end = time.time()
print("run time is %s" % (end - start))
return res # 我们装饰器的核心是wrapper函数,只需要在wrapper函数中提供返回值即可
return wrapper @outter # @+装饰器名称,可以快速完成 pint1 = outter(print1)的函数赋值
def print1(name):
print(name)
time.sleep(1)
return 111

可以看到,我们从outter()和wrapper()向内层函数进行了传参,outter函数传入的是我们所要修饰的函数对象,wrapper则是将参数原封不动的传给了修饰的函数对象,如果现在有个新的需求:

  • 设计一个装饰器,为函数增加认证功能,其中登陆的认证来源有多个(file、mysql)

首先看一下无参装饰器能否实现:

# 认证功能装饰器
def outter(func):
def wrapper(*args,**kwargs):
name = input("请输入用户名>>>")
pwd = input("请输入密码>>>")
if mode == "1": # 需要通过mode进行判断信息来源,但是没有参数传入
if name == "zhangda" and pwd == "123":
print("登陆成功!")
res1 = func(*args,**kwargs)
return res1
else:
print("用户名或者密码错误")
elif mode == "2": #需要通过mode进行判断信息来源,但是没有参数传入
         print("认证来源:mysql")
     else:
         print("未知来源")
return wrapper

我们需要传入一个mode参数进行判断,但是wrapper()需要给func()原样传入参数,所以不能添加mode,外层的outter()可以为mode传参,但是由于python语法糖限制,里面除了func外,不能添加其他的参数,那这个mode参数怎么传入呢?

我们可以在最外层函数进行传入

# 认证功能装饰器
def outter(func):
mode = "file"
def wrapper(*args,**kwargs):
name = input("请输入用户名>>>")
pwd = input("请输入密码>>>")
if mode == "1":
if name == "zhangda" and pwd == "123":
print("登陆成功!")
res1 = func(*args,**kwargs)
return res1
else:
print("用户名或者密码错误")
elif mode == "2":
print("认证来源:mysql")
else:
print("未知来源")
return wrapper @outter
def f1():
print("hhhhh")

但是这样每次使用装饰器都需要修改,并且不能同时满足多个信息来源需求,回想之前给函数传参的两种方法,既然直接传参行不通,我们想到了利用闭包函数特性,再包一层函数,这就是有参装饰器

def outter2(mode="file"):
def outter(func):
def wrapper(*args,**kwargs):
name = input("Username:").strip()
pwd = input("Password:").strip() if mode == "file":
if name == "egon" and pwd == "123":
print("登陆成功!")
res = func(*args,**kwargs)
return res
else:
print("登陆失败!")
elif mode == "mysql":
print("基于mysql认证!")
else:
print("未知来源认证!")
return wrapper
return outter @outter2("mysql")
def index(name):
print("{}".format(name)) index("hhhhhh")

在这里将之前的装饰器outter看做一个整体,对他进行装饰,并且最外层函数可以传入任意参数的,有参装饰的语法糖:@函数名(参数1,参数2......)

二、装饰器wraps

我们知道,装饰器其实就是定义了一个闭包函数,包含了原来的函数与功能并且添加新功能,最后通过函数对象赋值实现了全局调用,但是函数内存地址已经改变,只是名称看着一样

def outter(func):
def wrapper(*args,**kwargs):
"""wrapper注释"""
print("11111")
res = func(*args,**kwargs)
return res
return wrapper @outter
def index():
"""index注释"""
print("2222") print(index.__name__) # 打印函数的名称
print(index.__doc__) # 打印函数的注释
print(index) # 打印函数内存地址

运行结果如下:

index只是披着index外衣的wrapper函数,他的本质其实是wrapper,这不会影响函数的原始使用,但是如果有需求,希望这些属性看着不变呢?

def outter(func):
def wrapper(*args,**kwargs):
"""wrapper注释"""
print("11111")
res = func(*args,**kwargs)
return res
wrapper.__name__ = func.__name__
wrapper.__doc__ = func.__doc__
return wrapper @outter
def index():
"""index注释"""
print("2222") print(index.__name__) # 打印函数的名称
print(index.__doc__) # 打印函数的注释
print(index) # 打印函数内存地址

我们可以通过将wrapper的名称等信息重新复制,将原函数的信息赋值给wrapper即可:

其实python内置了一个库functools,提供了装饰器实现这一功能,用法如下:

from functools import wraps

def outter(func):
@wraps(func)
def wrapper(*args,**kwargs):
"""wrapper注释"""
print("11111")
res = func(*args,**kwargs)
return res
# wrapper.__name__ = func.__name__
# wrapper.__doc__ = func.__doc__
return wrapper @outter
def index():
"""index注释"""
print("2222") print(index.__name__) # 打印函数的名称
print(index.__doc__) # 打印函数的注释
print(index) # 打印函数内存地址

三、总结

装饰器模板

  • 无参装饰器
def outter(func):
def wrapper(*args,**kwargs):
res = func(*args,**kwargs)
return res
return wrapper
  • 有参修饰器
def outter2(x):
def outter(func):
def wrapper(*args,**kwargs):
print(x)
res = func(*args,**kwargs)
return res
return wrapper
return outter @outter2("1")
def f1():
print("2")

python函数之有参装饰器的更多相关文章

  1. Python函数07/有参装饰器/多个装饰器装饰一个函数

    Python函数07/有参装饰器/多个装饰器装饰一个函数 目录 Python函数07/有参装饰器/多个装饰器装饰一个函数 内容大纲 1.有参装饰器 2.多个装饰器装饰一个函数 3.今日总结 3.今日练 ...

  2. Python 函数修饰符(装饰器)的使用

     Python 函数修饰符(装饰器)的使用 1.  修饰符的来源修饰符是一个很著名的设计模式,经常被用于有切面需求的场景,较为经典的有插入日志.性能测试.事务处理等. 修饰符是解决这类问题的绝佳设计, ...

  3. python学习Day14 带参装饰器、可迭代对象、迭代器对象、for 迭代器工作原理、枚举对象、生成器

    复习 函数的嵌套定义:在函数内部定义另一个函数 闭包:被嵌套的函数 -- 1.外层通过形参给内层函数传参 -- 2.返回内部函数对象---->  延迟执行, 开放封闭原则: 功能可以拓展,但源代 ...

  4. python 函数名 、闭包 装饰器 day13

    1,函数名的使用. 函数名是函数的名字,本质就是变量,特殊的变量.函数名()加括号就是执行此函数. 1,单独打印函数名就是此函数的内存地址. def func1(): print(555) print ...

  5. 第十七篇 Python函数之闭包与装饰器

    一. 装饰器 装饰器:可以拆解来看,器本质就是函数,装饰就是修饰的意思,所以装饰器的功能就是为其他函数添加附加功能. 装饰器的两个原则: 1. 不修改被修饰函数的源代码 2. 不修改被修饰函数的调用方 ...

  6. python函数之闭包函数与无参装饰器

    一.global与nonlocal #global x = 1 def f1(): global x # 声明此处是全部变量x x = 2 print(x) f1() # 调用f1后,修改了全局变量x ...

  7. python函数与模块(装饰器,文件处理,迭代器等)

    os模块 os.system('命令') 利用python调用系统命令,命令可以是以列表或者元组内的元素形式* res import os res=os.system('ipconfig') prin ...

  8. python中的无参装饰器和有参装饰器

    python中的无参装饰器和有参装饰器 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 装饰器特点: 1>.开放封闭原则,即对扩展是开放的,对修改时封闭的: 2>.装饰器 ...

  9. python函数超时,用装饰器解决 func_timeout

    https://zhuanlan.zhihu.com/p/39743129 https://www.jianshu.com/p/a7fc98c7af4d https://ixyzero.com/blo ...

随机推荐

  1. 如何使用 js 检测控制台被用户打开了

    如何使用 js 检测控制台被用户打开了 js solutions 监听 F12 事件 监听键盘快捷键组合 Ctrl + Shift + I Option + Command + I Object.to ...

  2. Regular Expression & rgb2hex

    Regular Expression & rgb2hex regex // 颜色字符串转换 function rgb2hex(sRGB = 'rgb(255, 255, 255)') { co ...

  3. IM & RTC

    IM & RTC 即时通信(IM) & 实时通信(RTC) 场景 即时通信(可靠性高,延时高) 场景包括文字聊天.语音消息发送.文件传输.音视频播放等; 发短信 实时通信(可靠性低,延 ...

  4. javascript algorithm visualization

    javascript algorithm visualization javascript算法可视化 https://algorithm-visualizer.org https://github.c ...

  5. vue 使用stylus

    λ yarn add stylus stylus-loader --dev <style scoped lang="stylus"> div color #ff4488 ...

  6. 5分钟入门websocket

    5 个步骤快速掌握消息发送和接收 获取您的 appkey 先注册一个irealtime账号,然后登录到后台管理端,创建一个免费应用,就能得到您的 appkey.点击注册 各种前端生态端集成 ireal ...

  7. django学习-2.urls.py和view.py的相关知识点

    1.URL函数简单解析 1.1.url() 函数可以接收四个参数,分别是两个必选参数:regex.view,和两个可选参数:kwargs.name. def url(regex, view, kwar ...

  8. Java开发工程师最新面试题库系列——Spring部分(附答案)

    Spring Spring框架是什么? 答:Spring是轻量级的面向切面和控制反转的框架.初代版本为2002年发布的interface21,Spring框架是为了解决企业级应用开发的复杂性的出现的, ...

  9. 【Notes】现代图形学入门_02

    跟着闫令琪老师的课程学习,总结自己学习到的知识点 课程网址GAMES101 B站课程地址GAMES101 课程资料百度网盘[提取码:0000] 光栅化 着色(Shading) 在图形学中,着色的定义可 ...

  10. 基于docker部署skywalking实现全链路监控

    一.概述 简介 skywalking是一个开放源码的,用于收集.分析,聚合,可视化来自于不同服务和本地基础服务的数据的可观察的平台,skywalking提供了一个简单的方法来让你对你的分布式系统甚至是 ...