Python的装饰器用法学习笔记


Posted in Python onJune 24, 2016

在python中常看到在定义函数是使用@func. 这就是装饰器, 装饰器是把一个函数作为参数的函数,常常用于扩展已有函数,即不改变当前函数状态下增加功能.

def run():
  print "I'm run."

我有这么一个函数, 我想知道这个函数什么时候开始什么时候结束. 我应该这么写

def run():
  print time.ctime()
  print "I'm run."
  print time.ctime()

但是如果不允许修改函数的话就需要装饰器了

def count(func):
  def wrapper():
    print time.ctime()
    ret = func()
    print time.ctime()
    return ret
  return wrapper

@count
def run():
  print "I'm run."

      # print '2015-4-10'

eg:

def now():
  print '2015-4-10'
f = now
f()

 

函数有一个__name__ 对象 可通过 dir(func) func为定义的函数名

now.__name__    # print 'now'
f.__name__     # print 'now'

print f       # print '<function now at 0x000000000213A908>'
print now      # print '<function now at 0x000000000213A908>'

我们通过装饰器打印log日志

def log(func):
  def wrapper(*args, **kwargs):
    print "call %s()" % func.__name__
    return func(*args, **kwargs)
  return wrapper

@log
def now():
  print '2015-4-10'

now()        # print 'call now()'

其实装饰器修饰函数相当于, now = log(now) 也就是装饰器函数把被修饰的函数当参数后赋给同名的变量

functools.wraps 函数

当我们使用了装饰器后now的__name__值发生了改变

# 没有使用前
now.__name__    # print 'now'
# 使用后
now.__name__    # print 'wrapper'

当我们使用装饰器前,now.__name__使用的是当前now函数,但使用后 now这个函数其实是 log(now) 也就是log函数的返回值也就是被包裹的wrapper. 解决方法是functools.wraps函数.

装饰闭包, 使用前得调用 import functools

def log(func):
  @functools.wraps(func)
  def wrapper(*args, **kwargs):
    ...

带参数的装饰器

如果decorator需要传入参数, 那就需要在写一个返回decorator的高阶函数. 写出来更复杂.

def login(level):
  def _deco(func):
    def wrapper(*args, **kwargs):
      if level >= 5:
        print '用户 VIP 等级 %d' % int(level-5)
      else:
        print '用户 ?潘 等级 %d' % abs(level-5)
      return func(*args, **kwargs)
    return wrapper
  return _deco

@login(5)
def user(username):
  print 'welcome, %s' % username

# 用户vip 等级0
# welcome, mink
user('mink')

带参数的decorator等于func = 装饰器函数(装饰器参数)(func)

装饰器类

通过类的__call__可以想使用函数一样使用类

class A(object):
  def __init__(self, func):
    self.func = func

  def __call__(self):
    return self.func() ** 2

@A
def foo():
  return 10

print foo()   # print 100
Python 相关文章推荐
Python内置数据类型详解
Aug 18 Python
分析Python编程时利用wxPython来支持多线程的方法
Apr 07 Python
微信跳一跳自动运行python脚本
Jan 08 Python
python基于TCP实现的文件下载器功能案例
Dec 10 Python
Python lxml模块的基本使用方法分析
Dec 21 Python
python实现高斯投影正反算方式
Jan 17 Python
PyQt5高级界面控件之QTableWidget的具体使用方法
Feb 23 Python
python画图常规设置方式
Mar 05 Python
解决windows下python3使用multiprocessing.Pool出现的问题
Apr 08 Python
pandas 像SQL一样使用WHERE IN查询条件说明
Jun 05 Python
安装Anaconda3及使用Jupyter的方法
Oct 27 Python
Python进阶学习之带你探寻Python类的鼻祖-元类
May 08 Python
Python的网络编程库Gevent的安装及使用技巧
Jun 24 #Python
深入解析Python编程中super关键字的用法
Jun 24 #Python
深入了解Python数据类型之列表
Jun 24 #Python
Python实现信用卡系统(支持购物、转账、存取钱)
Jun 24 #Python
Python提取Linux内核源代码的目录结构实现方法
Jun 24 #Python
Linux上安装Python的PIL和Pillow库处理图片的实例教程
Jun 23 #Python
尝试用最短的Python代码来实现服务器和代理服务器
Jun 23 #Python
You might like
php的memcached客户端memcached
2011/06/14 PHP
单台服务器的PHP进程之间实现共享内存的方法
2014/06/13 PHP
php之Smarty模板使用方法示例详解
2014/07/08 PHP
php实现字符串反转输出的方法
2015/03/14 PHP
PHP读取zip文件的方法示例
2016/11/17 PHP
翻译整理的jQuery使用查询手册
2007/03/07 Javascript
javascript与CSS复习(《精通javascript》)
2010/06/29 Javascript
使用jquery制作弹出框效果
2015/04/03 Javascript
jquery实现表单输入时提示文字滑动向上效果
2015/08/10 Javascript
详解JS中Array对象扩展与String对象扩展
2016/01/07 Javascript
Bootstrap表单布局样式代码
2016/05/31 Javascript
js改变透明度实现轮播图的算法
2020/08/24 Javascript
jQuery实现带遮罩层效果的blockUI弹出层示例【附demo源码下载】
2016/09/14 Javascript
Javascript中call,apply,bind方法的详解与总结
2016/12/12 Javascript
js Canvas实现的日历时钟案例分享
2016/12/25 Javascript
js 单引号替换成双引号,双引号替换成单引号的实现方法
2017/02/16 Javascript
JS与jQuery实现子窗口获取父窗口元素值的方法
2017/04/17 jQuery
微信小程序如何获取用户手机号
2018/01/26 Javascript
解决Mac node版本升级失败的问题
2018/05/16 Javascript
微信jssdk逻辑在vue中的运用详解
2018/11/14 Javascript
js实现图片放大并跟随鼠标移动特效
2019/01/18 Javascript
vue + elementUI实现省市县三级联动的方法示例
2019/10/29 Javascript
在vue中created、mounted等方法使用小结
2020/07/21 Javascript
[43:33]EG vs Spirit Supermajor 败者组 BO3 第一场 6.4
2018/06/05 DOTA
图文详解WinPE下安装Python
2016/05/17 Python
Python实现读取并保存文件的类
2017/05/11 Python
Python的UTC时间转换讲解
2019/02/26 Python
美国卡车、吉普车和SUV零件网站:4 Wheel Parts
2016/11/24 全球购物
俄罗斯家居用品购物网站:Евродом
2020/11/21 全球购物
营业员演讲稿
2013/12/30 职场文书
教师学习培训邀请函
2014/02/04 职场文书
学校联谊活动方案
2014/02/15 职场文书
高中军训感言1000字
2014/03/01 职场文书
教师个人年度总结
2015/02/11 职场文书
初中家长意见
2015/06/03 职场文书
Python List remove()实例用法详解
2021/08/02 Python