python语言程序设计基础教程答案(Python装饰器实例讲解(一))
Python装饰器实例讲解(一)
多种角度讲述这个知识 ,这是个系列文章
但前后未必有一定的顺承关系
部分参考网络
本文以一个小案例引出装饰器的一些特点 ,不涉及理论 ,后面再谈
案例
写一个代码来求一个数是否是质数
def is_prime(x): if x == 2 : return True elif x == 1 or x % 2 == 0 : return False for i in range(3, int(x ** 0.5) + 1, 2): if x % i == 0: return False return True写个代码来计算某个数值范围内有多少个质数
def get_prime_nums(): return len(list(filter(is_prime,range(2,50000))))换一下 ,我们不是要学这个 ,我们要学装饰器
def get_prime_nums(): from time import time start_time = time() prime_nums = 0 for num in range(2,50000): if is_prime(num): prime_nums = prime_nums + 1 end_time = time() print(f统计花了{end_time-start_time}时间) print(f一共有{prime_nums}个质数) get_prime_nums() # 统计花了0.025316476821899414时间 # 一共有5133个质数你在这里会发现一个潜在的需求 ,可能不光是你这么一个函数有统计时间的需求 ,其他函数一样有 ,但现在这种处理方法可能要在每个目标函数上去加那段时间处理的代码 ,非常麻烦,那有没有好的做法呢?答案就是装饰器 。
装饰器
改造(对比下跟之前的区别)
获取质数个数函数 ,不需要统计时间
def get_prime_nums(): prime_nums = 0 for num in range(2,50000): if is_prime(num): prime_nums = prime_nums + 1 print(f一共有{prime_nums}个质数)写一个装饰器的函数(不用管为何这么写 ,以后会详细说明)
def count_time(func): def wrapper(): from time import time start_time = time() func() end_time = time() print(f统计花了{end_time-start_time}时间) return wrapper给要加时间的函数套上这个装饰器
@count_time def get_prime_nums(): prime_nums = 0 ... # 不重复了再次执行get_prime_nums()效果跟之前是一样的
同样的你可以将这个装饰器运用到其他函数上去
@count_time def get_odd_nums(): odd_nums = 0 for num in range(2,50000): if num % 2 == 1: odd_nums = odd_nums + 1 print(f一共有{odd_nums}个奇数) get_odd_nums()完了吗,没有 。可能性还有很多 ,主要是被装饰函数的变化导致了装饰器本身要随之适应变化。
装饰器改造一
如果被装饰的函数有返回值呢?
@count_time def get_prime_nums(): prime_nums = 0 for num in range(2,50000): if is_prime(num): prime_nums = prime_nums + 1 return prime_nums此时你直接调用函数 ,而不改造装饰器的话,是无法得到这个数量的
get_prime_nums() # 统计花了0.032898664474487305时间 print(get_prime_nums()) # 统计花了0.039182424545288086时间 # None改造装饰器
如何改造呢?你应该要去理解装饰器的运行原理(没那么复杂 ,但我们这个课不深入 ,仅作为案例给你展示)
def count_time(func): def wrapper(): from time import time start_time = time() result = func() # 改动1: 用一个变量来接收func()的返回 end_time = time() print(f统计花了{end_time-start_time}时间) return result # 改动2: return出去 return wrapper此时你再执行
print(get_prime_nums()) # 统计花了0.054421424865722656时间 # 5133 就能看到这个返回值了完了吗?还没有 ,如果我们的被装饰函数有参数呢?
装饰器改造二
你的被装饰函数存在参数
@count_time def get_prime_nums(end): prime_nums = 0 for num in range(2,end): if is_prime(num): prime_nums = prime_nums + 1 return prime_nums print(get_prime_nums(50000))其实在IDE中get_prime_nums(50000)就会提示你意外实参
执行结果
Traceback (most recent call last): File "...\demo.py", line 37, in <module> print(get_prime_nums(50000)) TypeError: wrapper() takes 0 positional arguments but 1 was given这是初学者最困惑的地方了 ,等我们讲了原理(或者说诀窍)你应该就非常清楚为何会这样报错了
怎么修改呢?
def count_time(func): def wrapper(*args): # 改动1: 增加一个不定参数 from time import time start_time = time() result = func(*args) # func也增加 end_time = time() print(f统计花了{end_time-start_time}时间) return result return wrapper再次执行 ,就ok了
print(get_prime_nums(50000)) # 统计花了0.029825448989868164时间 # 5133但是要注意 ,这样的话 ,如果你的被装饰函数是之前的没有参数的情况 ,是会报错的
# 回到过去 @count_time def get_prime_nums(): prime_nums = 0 for num in range(2,50000): if is_prime(num): prime_nums = prime_nums + 1 return prime_nums print(get_prime_nums(50000))报错
Traceback (most recent call last): File "...\demo.py", line 37, in <module> print(get_prime_nums(50000)) File "...\demo.py", line 22, in wrapper result = func(*args) TypeError: get_prime_nums() takes 0 positional arguments but 1 was given 进程已结束 ,退出代码为 1但由于是*args,你改成多个参数倒是可以的
@count_time def get_prime_nums(start,end): prime_nums = 0 for num in range(start,end): if is_prime(num): prime_nums = prime_nums + 1 return prime_nums print(get_prime_nums(2,50000)) # 可以执行如果你这样调用
print(get_prime_nums(start=2,end=50000))报错
Traceback (most recent call last): File "...\demo.py", line 37, in <module> print(get_prime_nums(start=2,end=50000)) TypeError: wrapper() got an unexpected keyword argument start 进程已结束 ,退出代码为 1可以这样修改你的装饰器
def count_time(func): def wrapper(*args,**kwargs): # 加个关键字参数 from time import time start_time = time() result = func(*args,**kwargs) # 这样也要加 end_time = time() print(f统计花了{end_time-start_time}时间) return result return wrapper说在最后
这个案例是入门的 ,讲解了装饰器的一些简单使用 但,留了一些坑 ,你可能未必知道为何要这么修改 ,装饰器是怎么调度的等等 且听下回分解创心域SEO版权声明:以上内容作者已申请原创保护,未经允许不得转载,侵权必究!授权事宜、对本内容有异议或投诉,敬请联系网站管理员,我们将尽快回复您,谢谢合作!