在Python里面函数也是一个对象,而且函数对象可以被赋值给变量,所以,通过变量也能调用该函数,简单来说函数也是变量也可以作文函数的参数

>>> def funA():
... print('i an funA')
...   time.sleep(1)
>>> f = now
>>> f()
i am funA

函数对象有一个__name__属性,可以拿到函数的名字:

>>> funA.__name__
'funA'
>>> f.__name__
'funA'

当我们想计算这个函数的运行时间,可以再创一个函数:

def print_time():
start = time.time()
funA()
end = time.time()
print('耗时%s秒' % (end - start)) >>>print_time()

i am funA
耗时1.0043737888336182秒

这时调用的并不是funA()而是print_time(),要是我们只能调用funA又要输出他的函数运行时间,这该怎么办?这时就可以派出今天的主人公,deractor(装饰器),简单来说在一个函数里面再嵌套一个函数,在这之前再看一个版本:

import time
# 随便定义一个函数
def funA():
print('i am funA')
time.sleep(1) def log(func):
def wrapper():
start = time.time()
func()
end = time.time()
print('耗时%s秒' % (end - start))
return wrapper if __name__ == '__main__':
n = log(funA)
n()

输出结果为:

i am funA
耗时1.0012261867523193秒

依旧可以,但好像有点麻烦,我们可以在def funA():上面加上@log,有个小细节,log(func)必须要在funA()之前,不然@log找不到,总的代码贴上:

import time
# 随便定义一个函数
def log(func):
def wrapper():
start = time.time()
func()
end = time.time()
print('耗时%s秒' % (end - start))
return wrapper
@log
def funA():
print('i am funA')
time.sleep(1) if __name__ == '__main__':
funA()

运行结果和上面一样,这样看起来就好多了,当然一个函数不单单可以嵌套一个函数,可以多个看你的需求,不同的嵌套发挥不同的作用。一开始说的,每个函数有个自带的.__name__属性,在主函数加个print(funA().__name__)会发现输出的是wrapper,这是因为log里面return的是wrapper,在某些情况下我们需要根据函数所属名来判断条件,那装饰器就成为累赘了,还好有个module(模块)叫functools,导入它并在def wrapper():前面加上@functools.wraps(func),意思就是将func包起来,不让他指向别的函数,这下运行结果就正常了:

import time
import functools
# 随便定义一个函数
def log(func):
# 函数可以接受任意参数的调用
@functools.wraps(func)
def wrapper():
start = time.time()
func()
end = time.time()
print('耗时%s秒' % (end - start))
return wrapper
@log
def funA():
print('i am funA')
time.sleep(1) if __name__ == '__main__':
funA()
print(funA.__name__)

运行结果:

i am funA
耗时1.0033290386199951秒
funA

Python3.5:装饰器的使用的更多相关文章

  1. (转)Python3.5——装饰器及应用详解

    原文:https://blog.csdn.net/loveliuzz/article/details/77853346 Python3.5——装饰器及应用详解(下)----https://blog.c ...

  2. python3.7 装饰器

    #!/usr/bin/env python __author__ = "lrtao2010" #python3.7 装饰器 #装饰器 ''' 定义:本质就是一个函数,作用是为其他函 ...

  3. python3练习-装饰器

    在廖雪峰的官方网站学习装饰器章节时,初步理解类似与面向切面编程.记录一下自己的课后习题解法. 问题: 请编写一个decorator,能在函数调用的前后打印出'begin call'和'end call ...

  4. python3之装饰器

    1.装饰器 装饰器本质上是一个python函数,它可以让其他函数在不需要做任何代码变动的前提下增加额外功能,装饰器的返回值也是一个函数对象.它经常用于有切面需求的场景,比如:插入日志.性能测试.事务处 ...

  5. python3之装饰器修复技术@wraps

    普通函数 def f(): """ 这是一个用来测试装饰器修复技术的函数 """ print("哈哈哈") if __n ...

  6. Python3中装饰器的使用

    较为复杂的装饰器使用: user,passwd = 'hjc',111111 def auth(type): print('auth type:',type) def outwrapper(func) ...

  7. python3 使用装饰器,及函数作为参数

    #装饰import typesdef shucai(n): print('蔬菜价格7') if type(n)==types.FunctionType: return n()+7 return n+7 ...

  8. 【python3】装饰器

    参考文章: 理解Python装饰器(Decorator) 关键点: 写装饰器一定要搞定楚函数名后面带小括号和不带小括号的含义.带小括号,表示调用这个函数,而不带小括号,则表示的是该函数引用地址 简单装 ...

  9. python3中装饰器的用法总结

    装饰器预备知识点 1 函数赋值给一个变量 函数名可以像普通变量一样赋值给另一个变量. def test(): print("i am just a test function") ...

随机推荐

  1. Python 学习之路3

    接下来把剩下的实验一起写上去 实验2 写一个学生类,属性有学号,姓名,成绩(三门),方法有输出,求平均成绩. 设计思路: 1.         先写一个学生类,并向里面写一个求平均值和输出信息的方法. ...

  2. 机器学习实验一SVM分类实验

    一.实验目的和内容 (一)实验目的 1.熟悉支持向量机SVM(Support Vector Machine)模型分类算法的使用. 2.用svm-train中提供的代码框架(填入SVM分类器代码)用tr ...

  3. php代码中的细节问题

    本次主要谈及工作中关于注销功能中的路径设置及session的清除问题之前的调试一直忽略了session的功能,以至于每次使用__ROOT__/index.php/home/Student/logout ...

  4. eric6 中 designer 无法启动的解决办法

    1.安装环境:win10+python3.6+Eric6 2.问题:使用 pip install PyQt5 安装 PyQt5.9 版本后,发现 Eric6 中无法打开 designer.exe 工具 ...

  5. 浅谈canvas绘画王者荣耀--雷达图

    背景: 一日晚上下班的我静静的靠在角落上听着歌,这时"滴!滴!"手机上传来一阵qq消息.原来我人在问王者荣耀的雷达图在页面上如何做出来的,有人回答用canvas绘画.那么问题来了, ...

  6. 自己动手编写IOC框架(二)

    万事开头难,上篇已经起了一个头,之后的事情相对就简单了.上次定义了框架所需的dtd也就是规定了xml中该怎么写,有哪些元素.并且我们也让dtd和xml绑定在了一起,使dtd对xml的格式进行校验,并且 ...

  7. Uncaught TypeError: download is not a function at HTMLAnchorElement.onclick (index.html:25)

    前段时间调试html报了这样的一个错误 Uncaught TypeError: download is not a function     at HTMLAnchorElement.onclick ...

  8. codeforge免费下载账号 积分账号 共享账号

    codeforge网站下载代码很好,没有积分怎么办?那么多好的matlab代码,matlab程序,C,JAVA等等,都要充值啊!!! 请用下面共享的codeforge账号密码========> ...

  9. Cocos2d-x3.0 触摸事件

    TouchableSpriteTest auto winSize = Director::getInstance()->getWinSize(); Point origin = Director ...

  10. [Android] AutoCompleteTextView:自己主动完毕输入内容的控件(自己主动补全)

    AutoCompleteTextView是EditText的直接子类,与普通EditText的最大不同就是.在用户输入的过程中,能够列出可供选择的输入项.方便使用者. AutoCompleteText ...