一、装饰器
装饰器:本质就是函数,功能是为其它函数添加附加功能
装饰器的原则:
- 不修改被修饰函数的源代码
- 不修改被修饰函数的调用方式
装饰器的知识储备:
装饰器 = 高阶函数 + 函数嵌套 + 闭包
二、高阶函数
高阶函数的定义:
- 函数的接收参数是一个函数名
- 函数的返回值是一个函数名
- 满足上述条件任意一个都可以是高阶函数

import time def fun1(): time.sleep(0.5) print("hello") def computing_run_time(fun): """ 计算函数运行时间 :param fun: :return: """ start_time = time.time() fun() end_time = time.time() print("运行时间%s" % (end_time - start_time)) computing_run_time(fun1) """ 优点:在不修改函数源代码的前提下,给函数添加了额外的功能 缺点:改变了调用方式 """

import time def fun1(): time.sleep(0.5) print("hello") def computing_run_time(fun): """ 计算函数运行时间 :param fun: :return: """ fun() return fun fun1 = computing_run_time(fun1) fun1() """ 优点:没有改变函数的调用方式 缺点:不能为函数添加新的功能 """
注:仅仅是高阶函数不能满足装饰器的需求
三、函数嵌套和闭包
"""
闭包:首先必须是内部定义的函数,该函数包含对外部作用域而不是全局作用域名字的引用
定义:内部函数的代码包含对外部函数的代码的引用,但一定不是对全局作用域的引用
"""
def fun1():
print("fun1")
name = 1
def fun2():
print("fun2")
print(name)
def fun3():
print("fun3")
print(name)
fun3()
fun2()
fun1()
四、装饰器示例
无参装饰器
def outer(fun):
def wrapper():
fun()
return wrapper
加上参数
def outer(fun):
def wrapper(*args, **kwargs):
fun(*args, **kwargs)
return wrapper
简单装饰器
import time
def outer(func):
"""
计算程序的运行时间
:param func:
:return:
"""
def wrapper(*args, **kwargs):
start_time = time.time()
result = func(*args, **kwargs)
stop_time = time.time()
print("函数运行的时间为:%s" % (stop_time-start_time))
return result
return wrapper
@outer # 语法糖 等价于 : cal = outer(cal) 把wrapper函数的函数地址赋值给cal
def cal(list1):
result = 0
for i in list1:
time.sleep(0.1)
result += i
return result
res = cal(range(20))
print(res)
注:
函数cal的执行流程,先执行装饰器的outer函数,outer将函数的返回值赋值给变量cal,所以调用cal() 等价于 cal = outer(cal), cal() 等价于这两步
多层装饰器
装饰器中含有函数参数
def before(name):
print("before:%s" % name)
return "before"
def after(name):
print("after:%s" % name)
return "after"
# 外层的参数只是为了将值传递到里层
def outer(before_func, after_func):
def middle(main_func):
def wrapper(*args, **kwargs):
before_result = before_func(*args, **kwargs)
print("before_result:%s" % before_result)
main_result = main_func(*args, **kwargs)
print("main_result: %s" % main_result)
after_result = after_func(*args, **kwargs)
print("after_result: %s" % after_result)
return main_result
return wrapper
return middle
@outer(before, after)
def index(name):
print("index")
return name
result = index("aaa")
print(result)

