网站的建设需要考虑什么问题,做网站和软件的团队,茶叶营销策划方案,网页版游戏大全在线玩学会这12个Python装饰器#xff0c;让你的代码更上一层楼
Python 装饰器是个强大的工具#xff0c;可帮你生成整洁、可重用和可维护的代码。某种意义上说#xff0c;会不会用装饰器是区分新手和老鸟的重要标志。如果你不熟悉装饰器#xff0c;你可以将它们视为将函数作为输… 学会这12个Python装饰器让你的代码更上一层楼
Python 装饰器是个强大的工具可帮你生成整洁、可重用和可维护的代码。某种意义上说会不会用装饰器是区分新手和老鸟的重要标志。如果你不熟悉装饰器你可以将它们视为将函数作为输入并在不改变其主要用途的情况下扩展其功能的函数。装饰器可以有效提高你的工作效率并避免重复代码。本文我整理了项目中经常用到的 12 个装饰器值得每一个Python开发者掌握。 文章目录1. logger2. wraps3. lru_cache4. repeat5. timeit6. retry7. countcall8. rate_limited9. dataclass10. register11. property12. singledispatch结论1. logger
我们从最简单的装饰器开始手动实现一个可以记录函数开始和结束的装饰器。被修饰函数的输出结果如下所示
some_function(args)# ----- some_function: start -----
# some_function executing
# ----- some_function: end -----要实现一个装饰器首先要给装饰器起一个合适的名称这里我们给装饰器起名为logger。
装饰器本质上是一个函数它将一个函数作为输入并返回一个函数作为输出。 输出函数通常是输入的扩展版。 在我们的例子中我们希望输出函数用start和end语句包围输入函数的调用。
由于我们不知道输入函数都带有什么参数我们可以使用 *args 和 **kwargs 从包装函数传递它们。*args 和 **kwargs 允许传递任意数量的位置参数和关键字参数。
下面是logger装饰器的示例代码
def logger(function):def wrapper(*args, **kwargs):print(f----- {function.__name__}: start -----)output function(*args, **kwargs)print(f----- {function.__name__}: end -----)return outputreturn wrapperlogger函数可以应用于任意函数比如
decorated_function logger(some_function)上面的语句是正确的但Python 提供了更 Pythonic 的语法——使用 修饰符。因此更通常的写法是
logger
def some_function(text):print(text)some_function(first test)
# ----- some_function: start -----
# first test
# ----- some_function: end -----some_function(second test)
# ----- some_function: start -----
# second test
# ----- some_function: end -----2. wraps
此装饰器更新wrapper 函数使其看起来像一个原始函数并继承其名字和属性。
要了解 wraps 的作用以及为什么需要它让我们将前面写的logger装饰器应用到一个将两个数字相加的简单函数中。
下面的代码是未使用wraps装饰器的版本
def logger(function):def wrapper(*args, **kwargs):wrapper documentationprint(f----- {function.__name__}: start -----)output function(*args, **kwargs)print(f----- {function.__name__}: end -----)return outputreturn wrapperlogger
def add_two_numbers(a, b):this function adds two numbersreturn a b如果我们用__name__ 和 __doc__来查看被装饰函数add_two_numbers的名称和文档会得到如下结果
add_two_numbers.__name__
wrapperadd_two_numbers.__doc__
wrapper documentation输出的是wrapper函数的名称和文档。这是我们预期想要的结果我们希望保留原始函数的名称和文档。这时wraps装饰器就派上用场了。
我们唯一需要做的就是给wrapper函数加上wraps装饰器。
from functools import wrapsdef logger(function):wraps(function)def wrapper(*args, **kwargs):wrapper documentationprint(f----- {function.__name__}: start -----)output function(*args, **kwargs)print(f----- {function.__name__}: end -----)return outputreturn wrapperlogger
def add_two_numbers(a, b):this function adds two numbersreturn a b再此检查add_two_numbers函数的名称和文档我们可以看到该函数的元数据。
add_two_numbers.__name__
# add_two_numbersadd_two_numbers.__doc__
# this function adds two numbers3. lru_cache
lru_cache是Python内置装饰器可以通过from functools import lru_cache引入。lru_cache的作用是缓存函数的返回值当缓存装满时使用least-recently-usedLRU算法丢弃最少使用的值。
lru_cache装饰器适合用于输入输出不变且运行时间较长的任务例如查询数据库、请求静态页面或一些繁重的处理。
在下面的示例中我使用lru_cache来修饰一个模拟某些处理的函数。然后连续多次对同一输入应用该函数。
import random
import time
from functools import lru_cachelru_cache(maxsizeNone)
def heavy_processing(n):sleep_time n random.random()time.sleep(sleep_time)# 初次调用
%%time
heavy_processing(0)
# CPU times: user 363 µs, sys: 727 µs, total: 1.09 ms
# Wall time: 694 ms# 第二次调用
%%time
heavy_processing(0)
# CPU times: user 4 µs, sys: 0 ns, total: 4 µs
# Wall time: 8.11 µs# 第三次调用
%%time
heavy_processing(0)
# CPU times: user 5 µs, sys: 1 µs, total: 6 µs
# Wall time: 7.15 µs从上面的输出可以看到第一次调用花费了694ms因为执行了time.sleep()函数。后面两次调用由于参数相同直接返回缓存值因此并没有实际执行函数内容因此非常快地得到函数返回。
4. repeat
该装饰器的所用是多次调用被修饰函数。这对于调试、压力测试或自动化多个重复任务非常有用。
跟前面的装饰器不同repeat接受一个输入参数
def repeat(number_of_times):def decorate(func):wraps(func)def wrapper(*args, **kwargs):for _ in range(number_of_times):func(*args, **kwargs)return wrapperreturn decorate上面的代码定义了一个名为repeat的装饰器有一个输入参数number_of_times。与前面的案例不同这里需要decorate函数来传递被修饰函数。然后装饰器定义一个名为wrapper的函数来扩展被修饰函数。
repeat(5)
def hello_world():print(hello world)hello_world()
# hello world
# hello world
# hello world
# hello world
# hello world5. timeit
该装饰器用来测量函数的执行时间并打印出来。这对调试和监控非常有用。
在下面的代码片段中timeit装饰器测量process_data函数的执行时间并以秒为单位打印所用的时间。
import time
from functools import wrapsdef timeit(func):wraps(func)def wrapper(*args, **kwargs):start time.perf_counter()result func(*args, **kwargs)end time.perf_counter()print(f{func.__name__} took {end - start:.6f} seconds to complete)return resultreturn wrappertimeit
def process_data():time.sleep(1)process_data()
# process_data took 1.000012 seconds to complete6. retry
当函数遇到异常时该装饰器会强制函数重试多次。它接受三个参数重试次数、捕获的异常以及重试之间的间隔时间。
其工作原理如下
wrapper函数启动num_retrys次迭代的for循环。将被修饰函数放到try/except块中。每次迭代如果调用成功则中断循环并返回结果。否则休眠sleep_time秒后继续下一次迭代。当for循环结束后函数调用依然不成功则抛出异常。
示例代码如下
import random
import time
from functools import wrapsdef retry(num_retries, exception_to_check, sleep_time0):遇到异常尝试重新执行装饰器def decorate(func):wraps(func)def wrapper(*args, **kwargs):for i in range(1, num_retries1):try:return func(*args, **kwargs)except exception_to_check as e:print(f{func.__name__} raised {e.__class__.__name__}. Retrying...)if i num_retries:time.sleep(sleep_time)# 尝试多次后仍不成功则抛出异常raise ereturn wrapperreturn decorateretry(num_retries3, exception_to_checkValueError, sleep_time1)
def random_value():value random.randint(1, 5)if value 3:raise ValueError(Value cannot be 3)return valuerandom_value()
# random_value raised ValueError. Retrying...
# 1random_value()
# 57. countcall
countcall用于统计被修饰函数的调用次数。这里的调用次数会缓存在wraps的count属性中。
from functools import wrapsdef countcall(func):wraps(func)def wrapper(*args, **kwargs):wrapper.count 1result func(*args, **kwargs)print(f{func.__name__} has been called {wrapper.count} times)return resultwrapper.count 0return wrappercountcall
def process_data():passprocess_data()
process_data has been called 1 times
process_data()
process_data has been called 2 times
process_data()
process_data has been called 3 times8. rate_limited
rate_limited装饰器会在被修饰函数调用太频繁时休眠一段时间从而限制函数的调用速度。这在模拟、爬虫、接口调用防过载等场景下非常有用。
import time
from functools import wrapsdef rate_limited(max_per_second):min_interval 1.0 / float(max_per_second)def decorate(func):last_time_called [0.0]wraps(func)def rate_limited_function(*args, **kargs):elapsed time.perf_counter() - last_time_called[0]left_to_wait min_interval - elapsedif left_to_wait 0:time.sleep(left_to_wait)ret func(*args, **kargs)last_time_called[0] time.perf_counter()return retreturn rate_limited_functionreturn decorate该装饰器的工作原理是测量自上次函数调用以来所经过的时间并在必要时等待适当的时间以确保不超过速率限制。其中等待时间min_interval - elapsed这里min_intervalue是两次函数调用之间的最小时间间隔以秒为单位已用时间是自上次调用以来所用的时间。如果经过的时间小于最小间隔则函数在再次执行之前等待left_to_wait秒。 ⚠注意该函数在调用之间引入了少量的时间开销但确保不超过速率限制。 如果不想自己手动实现可以用第三方包名叫ratelimit。
pip install ratelimit使用非常简单只需要装饰被调用函数即可:
from ratelimit import limitsimport requestsFIFTEEN_MINUTES 900limits(calls15, periodFIFTEEN_MINUTES)
def call_api(url):response requests.get(url)if response.status_code ! 200:raise Exception(API response: {}.format(response.status_code))return response如果被装饰函数的调用次数超过允许次数则会抛出ratelimit.RateLimitException异常。要处理该异常可以将sleep_and_retry装饰器与limits装饰器一起使用。
sleep_and_retry
limits(calls15, periodFIFTEEN_MINUTES)
def call_api(url):response requests.get(url)if response.status_code ! 200:raise Exception(API response: {}.format(response.status_code))return response这样被装饰函数在再次执行之前会休眠剩余时间。
9. dataclass
Python 3.7 引入了dataclass装饰器将其加入到标准库用于装饰类。它主要用于存储数据的类自动生成诸如__init__ __repr__ __eq__ __lt____str__ 等特殊函数。这样可以减少模板代码并使类更加可读和可维护。
另外dataclass还提供了现成的美化方法可以清晰地表示对象将其转换为JSON格式等等。
from dataclasses import dataclass, dataclass
class Person:first_name: strlast_name: strage: intjob: strdef __eq__(self, other):if isinstance(other, Person):return self.age other.agereturn NotImplementeddef __lt__(self, other):if isinstance(other, Person):return self.age other.agereturn NotImplementedjohn Person(first_nameJohn, last_nameDoe, age30, jobdoctor,)anne Person(first_nameAnne, last_nameSmith, age40, jobsoftware engineer,)print(john anne)
# Falseprint(anne john)
# Trueasdict(anne)
#{first_name: Anne,
# last_name: Smith,
# age: 40,
# job: software engineer}10. register
如果你的Python脚本意外终止但你仍想执行一些任务来保存你的工作、执行清理或打印消息那么register在这种情况下非常方便。
from atexit import registerregister
def terminate():perform_some_cleanup()print(Goodbye!)while True:print(Hello)运行上面的代码会不断在控制台输出Hello点击Ctrl C强制终止脚本运行你会看到控制台输出Goodbye说明程序在中断后执行了register装饰器装饰的terminate()函数。
11. property
property装饰器用于定义类属性这些属性本质上是类实例属性的getter、setter和deleter方法。
通过使用property装饰器可以将方法定义为类属性并将其作为类属性进行访问而无需显式调用该方法。
如果您想在获取或设置值时添加一些约束和验证逻辑使用property装饰器会非常方便。
下面的示例中我们在rating属性上定义了一个setter对输入执行约束介于0和5之间。
class Movie:def __init__(self, r):self._rating rpropertydef rating(self):return self._ratingrating.setterdef rating(self, r):if 0 r 5:self._rating relse:raise ValueError(The movie rating must be between 0 and 5!)batman Movie(2.5)
batman.rating
# 2.5batman.rating 4
batman.rating
# 4batman.rating 10# ---------------------------------------------------------------------------
# ValueError Traceback (most recent call last)
# Input In [16], in cell line: 1()
# ---- 1 batman.rating 10
# Input In [11], in Movie.rating(self, r)
# 12 self._rating r
# 13 else:
# --- 14 raise ValueError(The movie rating must be between 0 and 5!)
#
# ValueError: The movie rating must be between 0 and 5!12. singledispatch
singledispatch允许函数对不同类型的参数有不同的实现有点像Java等面向对象语言中的函数重载。
from functools import singledispatchsingledispatch
def fun(arg):print(Called with a single argument)fun.register(int)
def _(arg):print(Called with an integer)fun.register(list)
def _(arg):print(Called with a list)fun(1) # Prints Called with an integer
fun([1, 2, 3]) # Prints Called with a list结论
装饰器是一个重要的抽象思想可以在不改变原始代码的情况下扩展代码如缓存、自动重试、速率限制、日志记录或将类转换为超级数据容器等。
装饰器的功能远不止于此本文介绍的12个常用装饰器只是抛砖引玉当你理解了装饰器思想和用法后可以发挥创造力实现各种自定义装饰器来解决具体问题。
最后给大家推荐一个很棒的装饰器列表里面记录了大量实用的、有趣的装饰器大家可以多多尝试使用。