本文实例讲述了python3.5装饰器原理及应用。分享给大家供大家参考,具体如下:
1、装饰器:
(1)本质:装饰器的本质是函数,其基本语法都是用关键字def去定义的。
(2)功能:装饰其他函数,即:为其他函数添加附加功能。
(3)原则:不能修改被装饰的函数的源代码,不能修改被装饰的函数的调用方式。即:装饰器对待被修饰的函数是完全透明的。
(4)简单应用:统计函数运行时间的装饰器
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
import time
#统计函数运行时间的砖装饰器
def timmer(func):
def warpper( * args, * * kwargs):
strat_time = time.time()
func()
stop_time = time.time()
print ( "the func run time is %s" % (stop_time - strat_time))
return warpper
@timmer
def test1():
time.sleep( 3 )
print ( "in the test1" )
test1()
|
运行结果:
in the test1
the func run time is 3.000171661376953
(5)实现装饰器知识储备:
a、函数即“变量”
b、高阶函数
c、函数嵌套
d、高阶函数+嵌套函数==》装饰器
2、装饰器知识储备——函数即“变量”
定义一个函数,相当于把函数体赋值给这个函数名。
python解释器如何回收变量:采用引用计数。当引用有没有了时(门牌号不存在),变量就被回收了。
函数的定义也有内存回收机制,与变量回收机制一样。匿名函数没有函数名,就会被回收。
变量的使用:先定义再调用,只要在调用之前已经存在(定义)即可;函数即“变量”,函数的使用是一样的。
函数调用顺序:其他的高级语言类似,python 不允许在函数未声明之前,对其进行引用或者调用
下面的两段代码运行效果一样:
1
2
3
4
5
6
7
8
9
10
11
12
13
|
def bar():
print ( "in the bar" )
def foo():
print ( "in the foo" )
bar()
foo()
#python为解释执行,函数foo在调用前已经声明了bar和foo,所以bar和foo无顺序之分
def foo():
print ( "in the foo" )
bar()
def bar():
print ( "in the bar" )
foo()
|
运行结果:
in the foo
in the bar
in the foo
in the bar
注意:python为解释执行,函数foo在调用前已经声明了bar和foo,所以bar和foo无顺序之分
原理图为:
3、装饰器知识储备——高阶函数
满足下列其中一种即可称之为高阶函数:
a、把一个函数名当做实参传递给另一个函数(在不修改被装饰函数的情况下为其添加附加功能)
b、返回值中包含函数名(不修改函数的调用方式)
(1)高阶函数示例:
1
2
3
4
5
6
|
def bar():
print ( "in the bar" )
def test1(func):
print (func) #打印门牌号,即内存地址
func()
test1(bar) #门牌号func=bar
|
运行结果:
<function bar at 0x00bcdfa8>
in the bar
(2)高阶函数的妙处——把一个函数名当做实参传递给另一个函数(在不修改被装饰函数的情况下为其添加附加功能)
1
2
3
4
5
6
7
8
9
10
11
12
13
|
import time
def bar():
time.sleep( 3 )
print ( "in the bar" )
#test2在不修改被修饰函数bar的代码时添加了附加的及时功能
def test2(func):
start_time = time.time()
func() #run bar
stop_time = time.time()
print ( "the func run time is %s " % (stop_time - start_time))
#调用方式发生改变,不能像原来的方法去调用被修饰的函数(所以不能实现装饰器的功能)
test2(bar)
#bar()
|
运行结果:
in the bar
the func run time is 3.000171661376953
(3)高阶函数的妙处——返回值中包含函数名(不修改函数的调用方式)
1
2
3
4
5
6
7
8
9
|
import time
def bar():
time.sleep( 3 )
print ( "in the bar" )
def test3(func):
print (func)
return func
bar = test3(bar)
bar() #run bar
|
运行结果:
<function bar at 0x00badfa8>
in the bar
4、装饰器知识储备——嵌套函数
1
2
3
4
5
6
7
|
#函数嵌套
def foo():
print ( "in the foo" )
def bar(): #bar函数具有局部变量的特性,不能在外部调用,只能在内部调用
print ( "in the bar" )
bar()
foo()
|
运行结果:
in the foo
in the bar
装饰器应用——模拟网站登录页面,访问需要认证登录页面
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
|
#!/usr/bin/env python
# -*- coding:utf-8 -*-
# author:zhengzhengliu
#模拟网站,访问页面和部分需要登录的页面
import timer
user,passwd = "liu" , "liu123"
def auth(func):
def wrapper( * args, * * kwargs):
username = input ( "username:" ).strip()
password = input ( "password:" ).strip()
if username = = user and password = = passwd:
print ( "\033[32;1muser has passed authentication!\033[0m" )
res = func( * args, * * kwargs)
print ( "-----after authentication---" )
return res
else :
exit( "\033[31;1minvalid username or password!\033[0m" )
return wrapper
def index():
print ( "welcome to index page!" )
@auth
def home():
print ( "welcome to index home!" )
return "from home"
@auth
def bbs():
print ( "welcome to index bbs!" )
#函数调用
index()
print (home())
bbs()
|
运行结果:
welcome to index page!
username:liu
password:liu123
user has passed authentication!
welcome to home page!
-----after authentication---
from home
username:liu
password:liu123
user has passed authentication!
welcome to bbs page!
-----after authentication---
装饰器带参数
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
|
#!/usr/bin/env python
# -*- coding:utf-8 -*-
# author:zhengzhengliu
#模拟网站,访问页面和部分需要登录的页面,多种认证方式
import timer
user,passwd = "liu" , "liu123"
def auth(auth_type):
print ( "auth func:" ,auth_type)
def outer_wrapper(func):
def wrapper( * args, * * kwargs):
print ( "wrapper func args:" , * args, * * kwargs)
if auth_type = = "local" :
username = input ( "username:" ).strip()
password = input ( "password:" ).strip()
if username = = user and password = = passwd:
print ( "\033[32;1muser has passed authentication!\033[0m" )
#被装饰的函数中有返回值,装饰器中传入的参数函数要有返回值
res = func( * args, * * kwargs) #from home
print ( "-----after authentication---" )
return res
else :
exit( "\033[31;1minvalid username or password!\033[0m" )
elif auth_type = = "ldap" :
print ( "ldap...." )
return wrapper
return outer_wrapper
def index():
print ( "welcome to index page!" )
@auth (auth_type = "local" ) #利用本地登录 home = wrapper()
def home():
print ( "welcome to home page!" )
return "from home"
@auth (auth_type = "ldap" ) #利用远程的ldap登录
def bbs():
print ( "welcome to bbs page!" )
#函数调用
index()
print (home()) #wrapper()
bbs()
|
运行结果:
希望本文所述对大家python程序设计有所帮助。
原文链接:https://blog.csdn.net/loveliuzz/article/details/77853346