标签:**kwargs word 装饰器 语法 执行时间 details 阶乘 decorator []
“武林至尊,宝刀屠龙,号令天下,莫敢不从,倚天不出,谁与争锋”,这是神器。不过今天要说的python中的“神器”就没有这么厉害了,这里要说的“神器”其实就是名称里面带了个“器”的,如下:
现在遇到了这样一个问题需要解决:“有一个数字的列表,要求对该列表中的奇数乘以2,返回处理完成后的列表(不改变原来列表的顺序,仅对列表中的奇数乘以2)”,比较传统的方法可能会是这样的:
def double_odd_number_list(odd_number_list):
ret_value = []
for number in odd_number_list:
if number % 2:
ret_value.append(number * 2)
else:
ret_value.append(number)
return ret_value
测试数据:
if __name__ == ‘__main__‘:
test_odd_number_list = [1, 2, 3, 4, 5, 6, 7, 8, 9, 11, 24, 13, 28]
print double_odd_number_list(test_odd_number_list)
测试结果:
这样做对于新手来说比较容易理解,但代码不够精简,我们来看看列表解析器怎样完成上面的工作:
print [(number*2) if number % 2 else number
for number in test_odd_number_list]
代码说明:
test_odd_number_list还是上面定义的,测试结果如下:
上面讲的可能还满足不了你全部的需求,比如下面的情况:
列表解析器在生成的列表数据量很大时会占用很大的内存,它会一次性生成全部的数据保存到列表中并返回,无论是计算时间上还是内存消耗上,当数据量达到一定量级时都是无法接受的。一个改进方法就是使用生成器表达式,生成器表达式在语法上与列表解析唯一的区别就是讲[]换成了()。它会返回一个生成器,一次只计算一个值(延迟计算),如下图:
迭代器的详细内容可以参照我的另外一篇博客《python的迭代器》
谈到生成器,不得不谈到python的一个关键字yield,python中关于生成器的定义是:“A function which returns an iterator”,该函数与普通还是除了包含yield关键字外没有任何区别。yield关键字的作用是挂起函数的执行,下次调用时所有环境恢复到挂起时的状态从挂起的地方继续执行,一般都是放到循环里。我们来看一个求阶乘的例子:
有几点需要作出说明:
装饰器,装点门面用,此处的门面就是函数,用来装饰函数或者方法,在python中我们习惯性的把绑定到实例上的叫方法,未绑定的叫函数,也就是说在类里面第一个参数为self的就是方法,在模块中的就是函数。python中的装饰器有两种:带参数的装饰器,不带参数的装饰器
我们先从语法上相对简单的不带参数的装饰器讲起。我们在评估性能时,可能会精确到某个具体的函数执行一次需要花费的时间,新手最新想到的应该是写一个脚本,调用前和调用后分别获取一下当前时间,两者之差就是该函数执行所消耗的时间,代码可能是这样的:
def test(ret_value, sleep_time):
time.sleep(sleep_time)
return ret_value
if __name__ == ‘__main__‘:
start = datetime.datetime.utcnow()
test((1, 2, 3, 4, 5, 6), 3)
end = datetime.datetime.utcnow()
print (end - start).total_seconds()
代码容易理解,但如果有很多个函数都需要做这种测试呢?是不是就需要重复写很多次的:
start = datetime.datetime.utcnow()
end = datetime.datetime.utcnow()
print (end - start).total_seconds()
如果你想成为一个优秀的程序员,那么,尽你所能去避免做重复的事情,让你写的每一行代码发挥最大的价值,下面来看一个使用装饰器的例子:
import datetime
import time
__author__ = ‘Administrator‘
def time_counter(func):
def doc_func(*args, **kwargs):
start = datetime.datetime.utcnow()
ret = func(*args, **kwargs)
end = datetime.datetime.utcnow()
return ret, start, end, (end - start).total_seconds()
return doc_func
@time_counter
def test(ret_value, sleep_time):
time.sleep(sleep_time)
return ret_value
if __name__ == ‘__main__‘:
print test((1, 2, 3, 4, 5, 6), 3)
time_counter是一个装饰器,它跟普通函数的区别在于:
@time_counter def func(*args, **kwargs):
等价于func = time_counter(func)
@time_counter
def test(ret_value, sleep_time):
time.sleep(sleep_time)
return ret_value
当执行test函数时,会先将test函数对象作为参数传递到time_counter装饰器,也就是形式参数func,然后依次执行定义的doc_func中的全部操作,其中调用func的地方(ret = func(*args, **kwargs))会调用test函数,下面是上述代码的执行结果:
上面的装饰器,在使用时不需要传递任何参数,但有时这还不足以满足我们的需求,假设我们在一套测试框架中,测试用例可能有多种级别,完全测试时我们需要执行所有等级的测试用例,如果某次发布仅仅是修改了其中的一小块功能,对于未修改的部分,我们仅需要测试基本功能即可,例如级别大于等于2的。我们来看一下使用带参数的装饰器模拟实现该功能:
import datetime
import time
__author__ = ‘Administrator‘
def skip_unit_test(func):
print ‘skip function:%s‘ % func.func_name
def unit_test_filter(level):
def new_decorator(func):
def final_func(*args, **kwargs):
if level >= 2:
start = datetime.datetime.utcnow()
ret = func(*args, **kwargs)
end = datetime.datetime.utcnow()
print ret, start, end, (end - start).total_seconds()
else:
skip_unit_test(func)
return final_func
return new_decorator
@unit_test_filter(0)
def test_level_zero(ret_value, sleep_time):
time.sleep(sleep_time)
return ret_value
@unit_test_filter(1)
def test_level_one(ret_value, sleep_time):
time.sleep(sleep_time)
return ret_value
@unit_test_filter(2)
def test_level_two(ret_value, sleep_time):
time.sleep(sleep_time)
return ret_value
@unit_test_filter(3)
def test_level_three(ret_value, sleep_time):
time.sleep(sleep_time)
return ret_value
if __name__ == ‘__main__‘:
test_level_zero((1, 2, 3, 4, 5, 6), 3)
test_level_one((1, 2, 3, 4, 5, 6), 3)
test_level_two((1, 2, 3, 4, 5, 6), 3)
test_level_three((1, 2, 3, 4, 5, 6), 3)
看起来比较复杂,我们来总结一下带参数装饰器的特性:
@unit_test_filter(0)
等 标签:**kwargs word 装饰器 语法 执行时间 details 阶乘 decorator []
原文地址:http://www.cnblogs.com/wangjian8888/p/6266803.html