python 装饰器的应用
import time def test1():
print "hello\n"
print test1.__name__
def test2():
print "hello\n"
print test2.__name__
start = time.time()
test1()
end = time.time()
print "运行时间是:{}".format(end-start)
start = time.time()
test2()
end = time.time()
print "运行时间是:{}".format(end-start) 运行结果:
如上两段代码,我想计算方法的运行时间,这样写两次代码有点繁琐,而且观察每段代码都有重复的语句,print "hello\n",和后面的打印名字,那么,我们怎么写可以去掉方法冗余的代码?那么,就来介绍装饰器,首先,装饰器是在不改变源代码的基础上可以增加方法的功能。
首先python方法里面可以套方法,那么,我们可以装饰我们需要的方法。
import time def decorate(func):
def function():
start = time.time()
func()
end = time.time()
print "运行时间是:{}".format(end - start)
function() def test1():
print "hello"
print test1.__name__ def test2():
print "hello"
print test1.__name__ decorate(test1)
decorate(test2) 执行结果:
首先,我们要的方法是一定要执行的,只是执行的方式变了,而是把它封装在了另一个方法中,将方法名字作为参数,传给封装的方法(decorate),在我们的方法(func)前后,填入我们想实现的功能的语句(start = time.time(), end = time.time(),print "运行时间是:{}".format(end - start)),这只是实现了方法封装的思路,具体还不是这样子写。
首先,对原函数做了包装并返回了另外一个函数,额外添加了一些功能.
代码的改写,首选用@语法糖,将装饰器名字写在方法上面,装饰器内部需要返回方法,而不是直接执行方法,这是最简单的装饰器。
import time
def decorate(func):
def function():
start = time.time()
func()
end = time.time()
print "运行时间是:{}".format(end - start)
return function def hello(func):
start = time.time()
func()
end = time.time()
print "运行时间是:{}".format(end - start) @decorate
def test1():
print "hello"
print test1.__name__ @decorate
def test2():
print "hello"
print test2.__name__ test1()
test2()
但是有一个问题,如果被装饰的函数需要传入参数,那么这个装饰器就坏了。因为返回的函数并不能接受参数,你可以指定装饰器函数function接受和原函数一样的参数,比如:
import time def decorate(func):
def function(a,b):
start = time.time()
func(a,b)
end = time.time()
print "运行时间是:{}".format(end - start)
return function @decorate
def test2(a,b):
print "hello",a+b test2(3,7)
所以有一个问题,如果有的方法没有参数,有的方法有参数,有的方法参数数量不一致,类型不一致,难道要写很多个装饰器吗?python的语法很好的解决了这一个问题,那就是参数的处理,参数随着函数自已选择是不是有参。
import time def hello(func):
start = time.time()
func()
end = time.time()
print "运行时间是:{}".format(end - start) def decorate(func):
def function(*args,**kwargs):
print func.__name__
start = time.time()
func(*args,**kwargs)
end = time.time()
print "运行时间是:{}".format(end - start)
return function @decorate
def test1():
print "hello" @decorate
def test2(a,b):
print "hello",a+b test2(3,7)
test1() 运行结果:
更高级的装饰器:带参数的装饰器
import time def logging(level):
def decorate(func):
def function(*args, **kwargs):
print func.__name__,level
start = time.time()
func(*args, **kwargs)
end = time.time()
print "运行时间是:{}".format(end - start)
return function
return decorate @logging(level='INFO') #等价于 logging(level='INFO')(test3)
def test3():
pass test3() 运行结果:
类构造器:要了解类构造器,首先要了解__call__这个函数
__call__()
Python中的函数是一级对象。这意味着Python中的函数的引用可以作为输入传递到其他的函数/方法中,并在其中被执行。
而Python中类的实例(对象)可以被当做函数对待。也就是说,我们可以将它们作为输入传递到其他的函数/方法中并调用他们,正如我们调用一个正常的函数那样。而类中__call__()函数的意义正在于此。为了将一个类实例当做函数调用,我们需要在类中实现__call__()方法。也就是我们要在类中实现如下方法:def __call__(self, *args)。这个方法接受一定数量的变量作为输入。
假设a是Class类的一个实例。那么调用a.__call__()等同于调用a()。这个实例本身在这里相当于一个函数。
比如:
class Class(object):
def __call__(self, *args, **kwargs):
print "aa"
a=Class()
a()
执行结果:
aa
我们可以让类的构造函数__init__()接受一个函数,然后重载__call__()并返回一个函数,也可以达到装饰器函数的效果。
class Class(object):
def __init__(self,func):
self.func = func
def __call__(self, *args, **kwargs):
print "start"
self.func(*args, **kwargs)
print "end"
@Class def Ctest(something): print "say",something Ctest("love")
执行结果:
start
say love
end
带参数的类装饰器:
class Class(object):
def __init__(self,level):
self.level = level
def __call__(self, func): #接受函数
def wrapper(*args, **kwargs):
print "start"
func(*args, **kwargs)
print "end"
return wrapper #返回函数
@Class(level="INFO")
def Ctest(something):
print "say",something
Ctest("love")
执行结果:
start
say love
end
python 装饰器的应用的更多相关文章
- 关于python装饰器
关于python装饰器,不是系统的介绍,只是说一下某些问题 1 首先了解变量作用于非常重要 2 其次要了解闭包 def logger(func): def inner(*args, **kwargs) ...
- python装饰器通俗易懂的解释!
1.python装饰器 刚刚接触python的装饰器,简直懵逼了,直接不懂什么意思啊有木有,自己都忘了走了多少遍Debug,查了多少遍资料,猜有点点开始明白了.总结了一下解释得比较好的,通俗易懂的来说 ...
- Python 装饰器学习
Python装饰器学习(九步入门) 这是在Python学习小组上介绍的内容,现学现卖.多练习是好的学习方式. 第一步:最简单的函数,准备附加额外功能 1 2 3 4 5 6 7 8 # -*- c ...
- python 装饰器修改调整函数参数
简单记录一下利用python装饰器来调整函数的方法.现在有个需求:参数line范围为1-16,要求把9-16的范围转化为1-8,即9对应1,10对应2,...,16对应8. 下面是例子: def fo ...
- python 装饰器学习(decorator)
最近看到有个装饰器的例子,没看懂, #!/usr/bin/python class decorator(object): def __init__(self,f): print "initi ...
- Python装饰器详解
python中的装饰器是一个用得非常多的东西,我们可以把一些特定的方法.通用的方法写成一个个装饰器,这就为调用这些方法提供一个非常大的便利,如此提高我们代码的可读性以及简洁性,以及可扩展性. 在学习p ...
- 关于python装饰器(Decorators)最底层理解的一句话
一个decorator只是一个带有一个函数作为参数并返回一个替换函数的闭包. http://www.xxx.com/html/2016/pythonhexinbiancheng_0718/1044.h ...
- Python装饰器由浅入深
装饰器的功能在很多语言中都有,名字也不尽相同,其实它体现的是一种设计模式,强调的是开放封闭原则,更多的用于后期功能升级而不是编写新的代码.装饰器不光能装饰函数,也能装饰其他的对象,比如类,但通常,我们 ...
- Python装饰器与面向切面编程
今天来讨论一下装饰器.装饰器是一个很著名的设计模式,经常被用于有切面需求的场景,较为经典的有插入日志.性能测试.事务处理等.装饰器是解决这类问题的绝佳设计,有了装饰器,我们就可以抽离出大量函数中与函数 ...
- python装饰器方法
前几天向几位新同事介绍项目,被问起了@login_required的实现,我说这是django框架提供的装饰器方法,验证用户是否登录,只要这样用就行了,因为自己不熟,并没有做过多解释. 今天查看dja ...
随机推荐
- GitHub & OAuth 2.0 & JWT
GitHub & OAuth 2.0 & JWT https://www.rfcreader.com/#rfc6749 GitHub & OAuth https://www.b ...
- 二、kubernetes环境搭建
主要内容 1.环境准备(2主机) 2.安装流程 3.问题分析 4.总结 环境配置(2主机) 系统:CentOS 7.3 x64 网络:局域网(VPC) 主机: master:172.16.0.17 m ...
- Lodop文本项相对于文本框居中 两端对齐
Lodop中ADD_PRINT_TEXT默认内容是相对于文本框居左的,如果想要设置相对于文本框居中,可用如下语句.还有一种是两端对齐,可以让内容的两端阿和文本框的最左和最右端对齐,文本项内容布满文本框 ...
- vue的 v-for 循环中图片加载路径问题
先看一下产品需求,如下图所示, 产品要求图片和它的名称一一对应,本来是非常简单的需求,后台直接返回图片路径和名称,前台直接读取就可以了,但是我们没有存储图片的服务器,再加上是一个实验性的需求,图片需要 ...
- Redux 学习(1) ----- Redux介绍
Redux 有三个基本的原则: 1,单一状态树,redux 只使用一个javascript 对象来保存整个应用的状态. 状态树样式如下: const state = { count: 0 } 2,状态 ...
- codeforces570C
Replacement CodeForces - 570C 话说很久很久以前,孙悟空被压在了山下,很无聊.于是他找了一个只包含小写字母和字符"." 的字符串. 由于他比较无聊,他就 ...
- Linux系统下手把手完成无人值守安装服务
刚入职的运维新手经常会被要求去做一些安装操作系统的工作,如果按照用镜像光盘安装操作系统,效率会相当低下.那么如何提升效率,搭建出一套可以批量安装Linux系统的无人值守的安装系统? PXE+TFTP+ ...
- 【深入Java虚拟机】之一:Java内存模型
[深入Java虚拟机]之:Java内存区域与内存溢出 内存区域 Java虚拟机在执行Java程序的过程中会把他所管理的内存划分为若干个不同的数据区域.Java虚拟机规范将JVM所管理的内存分为以下几个 ...
- NMAP网络扫描工具的安装与使用
简介 NMAP是一款流行的网络扫描和嗅探工具也是一个强大的端口扫描类安全测评工具,被广泛应用在黑客领域做漏洞探测以及安全扫描,更多的nmap是一个好用的网络工具,在生产和开发中也经常用到,主要做端口开 ...
- JMeter——JMeter如何进行汉化
1.找到bin目录下的jmeter.properties文件 2.打开找到第37行,打开注释并将language=en改为language=zh_CN 3.重启



