# -*-coding=utf-8 -*-
#实现一个函数执行后计算执行时间的功能
__author__ = 'piay'
import time, functools def foo():
'''
定义一个普通函数
:return:
'''
print 'this is foo' foo() '''
这里如果我们需要查看函数执行时间,修改为:
''' def foo1():
start_time = time.clock()
print 'this is foo1'
end_time = time.clock()
print '执行时间为:', end_time - start_time foo1() '''
如果我们其他的函数也需要执行时间,或者这个函数不需要执行时间,那么我们就需要复制到其他的函数中去
这是一种最差的方法
''' def foo3():
print 'this is foo3' def timeit(func):
'''
我们可以考虑重新定义一个函数timeit,将foo的引用传递给他,
然后在timeit中调用foo并进行计时,这样,我们就达到了不改动foo定义的目的
:param func: 传入的函数
:return:
'''
start_time = time.clock()
func()
end_time = time.clock()
print 'used:', end_time - start_time timeit(foo3)
'''
这样写修改调用部分的代码。原本我们是这样调用的:foo3(),现在变成timeit(foo),这样的话,如果foo在N处都被调用了,
你就不得不去修改这N处的代码。或者更极端的,考虑其中某处调用的代码无法修改这个情况,比如:这个函数是你交给别人使用的。
''' '''
想想办法不修改调用的代码;如果不修改调用代码,也就意味着调用foo()需要产生调用timeit(foo)的效果。我们可以想到将timeit赋值给foo,
但是timeit似乎带有一个参数……想办法把参数统一吧!如果timeit(foo)不是直接产生调用效果,
而是返回一个与foo参数列表一致的函数的话……就很好办了,将timeit(foo)的返回值赋值给foo,然后,调用foo()的代码完全不用修改!
''' def foo4():
print 'this is foo4' # 定义一个计时器,传入一个,并返回另一个附加了计时功能的方法
def timeit4(func):
# 定义一个内嵌的包装函数,给传入的函数加上计时功能的包装
def wrapper():
start_time = time.clock()
func()
end_time = time.clock()
print 'used:', end_time - start_time # 将包装后的函数返回
return wrapper foo_1 = timeit(foo4)
'''
上面的代码就类似装饰器了,可以修改为如下:
''' @timeit4 # 定义上加上这一行与另外写foo = timeit(foo)完全等价
def foo5():
print 'this is foo5'
foo5() '''
-----------------------------------------------
使用functools.wraps(func)装饰器实现功能
'''
def timeit_3_for_wraps(func):
@functools.wraps(func)
def wrapper():
start=time.clock()
func()
end=time.clock()
print 'used:',end-start
return wrapper @timeit_3_for_wraps
def foo6():
print 'this is foo6'
foo6()

这里实现一个完整的判断是否带参数的装饰器:

# -*-coding=utf-8 -*-
__author__ = 'piay'
import functools, time
'''
一个函数执行前打印开始执行,执行完后打印执行完成,记录执行时间
''' def log(text):
if callable(text): # 参数如果是函数,说明装饰器不带参传过来,text是一个函数
@functools.wraps(text)
def wrapper(*args, **kwargs):
start = time.clock()
print '这是不带参数的装饰器,开始执行'
f = text(*args, **kwargs) #执行本身的函数 text()
end = time.clock()
print "结束执行:", end - start
return f # 返还原函数
return wrapper elif not callable(text): # text是参数,不是函数
def decarator(func):
@functools.wraps(func)
def warpper(*args, **kwargs):
start = time.clock()
print '这是不带参数的装饰器,开始执行,参数为:'+text
f = func(*args, **kwargs)
end=time.clock()
print "结束执行:",end-start
return f #返还原函数
return warpper
return decarator
else:
print '请检查是否正确' @log
def add1(x,y):
print x+y @log('')
def add2(x,y):
print x+y add1(1,2)
add2(2,3)

执行结果:

D:\Python27\python.exe D:/Python/functools_study/完整的装饰器.py
这是不带参数的装饰器,开始执行
3
结束执行: 5.08444509009e-05
这是不带参数的装饰器,开始执行,参数为:222
5
结束执行: 2.49333364995e-05

Process finished with exit code 0

python functools.wraps装饰器模块的更多相关文章

  1. python中functools.wraps装饰器的作用

    functools.wraps装饰器用于显示被包裹的函数的名称 import functools def node(func): #@functools.wraps(func) def wrapped ...

  2. 探究functools模块wraps装饰器的用途

    <A Byte of Python>17.8节讲decorator的时候,用到了functools模块中的一个装饰器:wraps.因为之前没有接触过这个装饰器,所以特地研究了一下. 何谓“ ...

  3. 关于functools模块的wraps装饰器用途

    测试环境:Python3.6.2 + win10 +  Pycharm2017.3 装饰器之functools模块的wraps的用途: 首先我们先写一个装饰器 # 探索functools模块wraps ...

  4. Python 中实现装饰器时使用 @functools.wraps 的理由

    Python 中使用装饰器对在运行期对函数进行一些外部功能的扩展.但是在使用过程中,由于装饰器的加入导致解释器认为函数本身发生了改变,在某些情况下——比如测试时——会导致一些问题.Python 通过  ...

  5. python学习笔记2-functools.wraps 装饰器

    wraps其实没有实际的大用处, 就是用来解决装饰器导致的原函数名指向的函数 的属性发生变化的问题: 装饰器装饰过函数func, 此时func不是指向真正的func,而是指向装饰器中的装饰过的函数 i ...

  6. python高级之装饰器

    python高级之装饰器 本节内容 高阶函数 嵌套函数及闭包 装饰器 装饰器带参数 装饰器的嵌套 functools.wraps模块 递归函数被装饰 1.高阶函数 高阶函数的定义: 满足下面两个条件之 ...

  7. Day04 - Python 迭代器、装饰器、软件开发规范

    1. 列表生成式 实现对列表中每个数值都加一 第一种,使用for循环,取列表中的值,值加一后,添加到一空列表中,并将新列表赋值给原列表 >>> a = [0, 1, 2, 3, 4, ...

  8. 第二篇:python高级之装饰器

    python高级之装饰器   python高级之装饰器 本节内容 高阶函数 嵌套函数及闭包 装饰器 装饰器带参数 装饰器的嵌套 functools.wraps模块 递归函数被装饰 1.高阶函数 高阶函 ...

  9. Python 中的装饰器

    说到装饰器是我们每个学Python人中的心痛. 装饰器作用:是用来装饰其他函数的,为其他函数添加新功能. 原则:1.不能改变被修饰函数的源代码. 2.不能修改被修饰函数的调用方式. 学装饰器前还需要了 ...

随机推荐

  1. Java虚拟机--字节码指令集

    1. 字节码指令集简介: Java虚拟机的指令由一个字节长度的,代表着某种特定操作含义的操作码(Opcode)以及跟随其后的零至多个代表此操作所需参数的操作数(Operands)所构成.虚拟机中许多指 ...

  2. Hexo学习笔记--常用命令及部署步骤

    参考资料: Hexo搭建Github静态博客:http://www.cnblogs.com/zhcncn/p/4097881.htmlHexo:独立博客新玩法:http://www.aips.me/h ...

  3. (摘)DataGuard物理standby管理 - 主备切换

    DataGuard物理standby管理 - 主备切换 Dataguard的切换分为两种,switchover和failover. switchover一般用于数据库或硬件升级,这时只需要较短时间中断 ...

  4. ultraedit删除空行(含空格,tab,制表符等怪字符)

    打开ultraedit,ctrl+r弹出替换对话框,点选启用正则表达式 在查找框输入 ^p^p 在替换框输入  ^p 仍有部分空行还在,继续处理: 查找框中输入:%[ ^t]++^p,注意^t之前有空 ...

  5. iptables 顺序

    -A INPUT -s 115.236.6.6/32 -p udp -m udp --dport 111 -j ACCEPT -A INPUT -s 10.175.197.98/32 -p udp - ...

  6. DOS批处理命令判断操作系统版本、执行各版本对应语句

    DOS批处理命令判断操作系统版本.执行各版本对应语句   昨天在家里试用  netsh interface ip set address 这些命令更改上网IP.DNS.网关等,今天将那些代码拿来办公室 ...

  7. cf494A Treasure

    A. Treasure time limit per test 2 seconds memory limit per test 256 megabytes input standard input o ...

  8. hdu5017:补题系列之西安网络赛1011

    补题系列之西安网络赛1011 题目大意:给定一个椭球: 求它到原点的最短距离. 思路: 对于一个椭球的标准方程 x^2/a^2 + y^2/b^2 +z^2/c^2=1 来说,它到原点的最短距离即为m ...

  9. IOS深入学习(12)之Archiving

    1 前言 本文介绍的是一个归档解档方法,也是编码和解码时候所做的事情,和如何进行,编码和归档其实就是将对象关系转化为字节流并且归档为特殊的文件,解码和解档是逆过程. 英文原文:http://blog. ...

  10. hdu 5392 Infoplane in Tina Town(数学)

    Problem Description There is a big stone with smooth surface in Tina Town. When people go towards it ...