适合初学者的Python装饰器的简易教程
前言
装饰器是Python编程语言中相当高级的一部分。就像大多数事情一样,一旦你掌握了它们的工作原理并使用了几次,它们就会变得非常简单明了,但是作为一个初学者,它们可能会有点让人望而生畏,很难理解。只有理解了它所解决的问题,你才能真正理解它。例如,我可以直接声明装饰器的定义:
decorator是一个函数,它将另一个函数作为参数并返回它的修改版本,以某种方式增强了它的功能。
如果您已经了解了decorator是什么,那么这个定义就非常清楚了,但是如果您不了解,那么可能就不太了解了。重要的是,这个定义本身并不能告诉您什么时候使用修饰符,或者没有修饰符Python会变得多么糟糕。
举例
我们将从一个假设的场景开始,并观察如果不使用decorator可能出现的问题。在你上班的第一天,你的老板找到你,让你写一个函数,这个函数将一个字符串转换成一个回文:一个向前和向后读取相同内容的字符串。
你可以这样写:
def make_palindrome(string):
"""Makes a palindromic mirror of a string."""
return string + string[::-1]
到目前为止一切顺利。一小时后,老板要求更多的函数:一个credits函数在任何字符串的末尾添加一个字符串,一个函数将字符串转换到另一个字符串中,还有一个函数在字符串中插入逗号。
你开始加入新的函数:
但问题出现了。老板看你的代码,并提醒你函数必须能够接受整数作为输入,并且它们应该被转换成字符串。他建议在每个函数的开头加上一行,检查输入是否为整数,如果是整数则进行转换。
这会让你士气低落——你必须把每个功能都检查一遍,然后在开始的时候加上一些类似这样的东西:
if isinstance(string, int):
string = str(string)
当我们有四个需要修改的函数时,这是可以的,但是如果我们有十个呢?让所有的功能都以相同的两行开始违背了神圣的“不要重复自己”的法律准则。
难道没有一种方法可以只修改所有这些函数而不添加额外代码吗?要了解如何做到这一点,让我们回过头来看看Python函数。尽管Python函数有特殊的语法,但它只是一个对象,就像字符串或列表一样。您可以检查它们的属性,将它们分配给新的变量,并且——至关重要的是——将它们作为参数传递给另一个函数。
例如,您可以使一个函数接受另一个函数,并检查它是否有任何关键字参数:
def func_has_kwargs(func):
return len(func.__defaults__) > 0
不要担心__defaults__如果你还没见过,这里的关键是,函数是另一个函数作为参数,检查是否有任何关键字参数(即如果__default__产权的长度大于0),否则,返回True,如果是这样,则返回False。
现在回到我们的问题之中。我们有三个精心设计的字符串操作函数,我们需要修改它们,使它们也接受整数。我们需要的是一个新函数——它将把我们现有的函数作为输入,并创建一个修改后的函数来检查整数。我们需要一个装饰函数:
让我们仔细看看这里发生了什么。accept_integers是我们的装饰函数——它接受一个函数作为输入,返回另一个函数作为输出。在它的主体中,它创建了一个新函数,该函数应该完成输入函数所做的所有事情,但是在开始时需要一个额外的步骤。如果您查看这个函数的主体,您可以看到它检查给定的字符串是否为整数,如果是整数则转换它,然后将这个字符串传递给原始函数。这里缺少一个步骤——我们需要实际使用这个装饰器:
标准形式
最后,值得指出的是,虽然上面的语法是完全有效的,但是Python以@符号的形式提供了快捷方式。可以将@accept_integers添加到任何函数的前面来修饰它:
这是将一个函数传递给另一个函数的另一种方式。在底层,当Python看到@符号时,它会为您执行decorator的调用。许多Python库都提供装饰器,以快速增强编写的函数,而不必输入大量重复的代码。
我们对装饰师和所有新编程特性的建议是,首先要学会识别使用该特性的情况——它能解决的问题,以及不使用它所带来的痛苦——然后再学习它是如何工作的。像往常一样,真正理解的唯一方法,就是自己编写一个。所以去做吧,后浪们。
参考: https://samireland.com/writing/decorators/
- CSS3制作心形头像
- CSS魔法堂:重拾Border之——不仅仅是圆角
- scala 学习笔记(01) 函数定义、分支、循环、异常处理、递归
- java之log4j的配置
- scala 学习笔记(02) 元组Tuple、数组Array、Map、文件读写、网页抓取示例
- scala 学习笔记(04) OOP(上)主从构造器/私有属性/伴生对象(单例静态类)/apply方法/嵌套类
- 使用jQuery封装实用函数
- scala 学习笔记(05) OOP(中)灵活的trait
- Web Fundamentsals学习1-Multiple-Screen-Site
- Vagrant使用
- java 中的异步回调
- 澳大利亚域名管理机构多年敲竹杠?
- ASP.NET中使用HttpWebRequest调用WCF
- scala 学习笔记(06) OOP(下)多重继承 及 AOP
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法