黑马程序员技术交流社区
标题:
【上海校区】装饰器
[打印本页]
作者:
不二晨
时间:
2019-3-8 10:16
标题:
【上海校区】装饰器
装饰器
装饰模式有很多经典的使用场景,例如插入日志、性能测试、事务处理等等,有了装饰器,就可以提取大量函数中与本身功能无关的类似代码,从而达到代码重用的目的。
装饰器引入
下面就简单举个例子:
一天,A程序员接到一个登入的需求,写了一个方法。
def login():
print('登入')
login()
# 输出
登入
1
2
3
4
5
6
7
8
9
突然,产品经理想加入一个登入事件。于是A程序员对方法进行了修改。
def login():
print('事件记录')
print('登入')
login()
# 输出
事件记录
登入
1
2
3
4
5
6
7
8
9
10
11
然后,A的老大看到了说,你这样违反了开闭原则,好好思考一下如何改进。后面A就改进,通过闭包的方式,定义了一个event函数。
# 定义一个event函数,记录事件
def event(func):
def inner():
print('事件记录')
func()
return inner
def login():
print('登入')
login = event(login) # 这个代码意思是,把定义的login函数作为变量传递给event作为参数。event函数的返回值在赋值login变量
login() # 调用login变量的方法
# 输出
事件记录
登入
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
这里有个比较难理解的地方就是,def定义了一个login函数,实际上它是一个login变量指向def定义的login函数的地址值。
eg:
def login():
print('登入')
print(login) # <function login at 0x000001D8D0543E18>
a = login # 将login函数的内存地址指向给a变量
a() # 调用a函数,实际上调用的是a指向的地址值的方法
# 输出
登入
1
2
3
4
5
6
7
8
9
10
这个例子就说明login是一个变量,这个变量指向login函数地址值。
理解了这个例子,那么login = event(login)就比较好解释了。
login = event(login)
等号左边的login是event()的返回值,是经过inner变量赋值,inner变量指向的是inner函数的地址值。
等号右边的login变量是指向login函数的地址值。作为参数传递给event函数。实际上是在inner函数中调用。
login()
调用login函数,因为login变量经过inner变量赋值,所以这时的login变量指向的是inner函数的地址值。
也就是说,login()调用,实际上调用的是inner函数,inner函数执行的顺序是:
print('事件记录')
func()
func参数是由event函数传入,也就是login = event(login)式子等号右边的login,这个login变量指向的是def定义login()。
所以输出的结果是:
事件记录
登入
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
装饰器语法糖
在Python中,可以使用@语法糖来精简装饰器的代码:
使用了@语法糖后,我们就不需要额外代码来给login重新赋值了。
eg:
def event(func):
def inner():
print('事件记录')
func()
return inner
@event # 实际上做了 login = event(login)操作
def login():
print('登入')
login()
# 输出
事件记录
登入
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
这里login()的@event,其实本质就是login = event(login),@做了一步赋值操作。当认清了这一点后,后面看带参数的装饰器就简单了。
被装饰的函数带参数
我们只带login()作为登入函数,实际上是有用户名和密码的,这个时候login()需要2个参数,那么怎么处理呢。
eg:
def event(func):
def inner(username, password):
print('事件记录')
func(username, password)
return inner
@event
def login(username, password):
print('username is %s password is %s,登入成功' %(username,password))
login('amy','123456')
# 输出
事件记录
username is amy password is 123456
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
如果这时候需要加上一个注册的方法,注册需要3个参数,怎么使用呢?这个时候需要使用可变参数*args, **kwargs。
eg:
def event(func):
def inner(*args, **kwargs):
print('事件记录')
func(*args, **kwargs)
return inner
@event
def login(username, password):
print('username is %s , password is %s,登入成功' % (username, password))
@event
def register(username, password, email):
print('username is %s , password is %s , email is %s,注册成功' % (username, password, email))
login('amy','123456')
# 输出
事件记录
username is amy , password is 123456 , 登入成功
register('anne', '654321' , 'abc@126.com')
# 输出
事件记录
username is amy , password is 123456 , email is
abc@126.com
, 注册成功
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
被修饰的函数带返回值
现在登入和注册成功后,需要有返回值,那么怎么处理,请看下面例子:
def event(func):
def inner(*args, **kwargs):
print('事件记录')
ret = func(*args, **kwargs)
return ret
return inner
@event
def login(username, password):
print('username is %s , password is %s' % (username, password))
return '登入成功'
@event
def register(username, password, email):
print('username is %s , password is %s , email is %s' % (username, password, email))
return '注册成功'
ret = login('amy','123456')
print(ret) # 输出 登入成功
ret = register('anne', '654321' , 'abc@126.com')
print(ret) # 输出 注册成功
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
如果函数没有返回值,怎么处理,函数没有返回值,其实返回的是None。上面的装饰器是比较通用的装饰器。
带参数的装饰器
eg:
def event(debug=True):
if (debug):
def deco(func):
def inner(*args, **kwargs):
print('事件记录')
return func(*args, **kwargs)
return inner
else:
def deco(func):
return func
return deco
@event() # 使用默认参数
def login(username, password):
print('username is %s , password is %s' % (username, password))
return '登入成功'
@event(debug=False) # debug参数设置成False,也可以直接写False
def register(username, password, email):
print('username is %s , password is %s , email is %s' % (username, password, email))
return '注册成功'
login('amy', '123456')
# 输出
事件记录
username is amy , password is 123456
register('anne', '654321', 'abc@126.com')
# 输出
username is anne , password is 654321 , email is
abc@126.com
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
@event加上参数,目的是可以灵活的控制函数的输出,如是否打开调试信息等。
如果@event加上参数,在login()函数定义之前就开始执行。我们可以看一下装饰器的执行顺序。
eg:
def event(debug=True):
print('event函数调用')
if (debug):
def deco(func):
print('deco函数调用')
def inner(*args, **kwargs):
print('inner函数调用')
# print('事件记录')
return func(*args, **kwargs)
return inner
else:
def deco(func):
print('deco函数调用')
return func
return deco
@event()
def login(username, password):
print('login函数调用')
# print('username is %s , password is %s' % (username, password))
return '登入成功'
# 在未调用函数时,运行,打印输出:
event函数调用
deco函数调用
login()
# 调用函数后,运行,输出:
event函数调用
deco函数调用
inner函数调用
login函数调用
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
也就是说,定义的event方法和deco方法在未调用时Python解释器就已经开始执行了。
执行步骤:
1.定义event函数
2.调用event函数,执行打印event函数调用语句,返回deco函数的引用
3.使用@event,执行打印deco函数调用语句,返回inner函数的引用
4.使用func进行装饰
5.调用login函数
6.执行inner函数调用语句
7.执行login函数调用语句
---------------------
【转载】
作者:张行之
来源:CSDN
原文:
https://blog.csdn.net/qq_33689414/article/details/78295628
版权声明:本文为博主原创文章,转载请附上博文链接!
作者:
不二晨
时间:
2019-3-11 15:24
奈斯,感谢分享
欢迎光临 黑马程序员技术交流社区 (http://bbs.itheima.com/)
黑马程序员IT技术论坛 X3.2