http://python.jobbole.com/80956/
本文是我称为「这不是魔法」系列的第一篇,我准备在里面展示一些热门开源包提供的友好API是如何通过它们各自语言的原始语法构造的。
本文我们先来说说Flask,深入探讨Flask如何实现在函数上方写“@app.route()”就能在因特网上输出函数的执行结果。
下面是Flask主页给我们的第一个例子,我们现在就由它入手,深入理解“@app.route()”是如何工作的。
1
2
3
4
5
|
app
=
Flask
(
__name__
)
@
app
.
route
(
"/"
)
def
hello
(
)
:
return
"Hello World!"
|
@app.route和其它装饰器
要想明白“@app.route()”的工作原理,我们首先需要看一看Python中的装饰器(就是以“@”开头的那玩意,下面接着函数定义)。
究竟什么是装饰器?没啥特别的。装饰器只是一种接受函数(就是那个你用“@”符号装饰的函数)的函数,并返回一个新的函数。
当你装饰一个函数,意味着你告诉Python调用的是那个由你的装饰器返回的新函数,而不仅仅是直接返回原函数体的执行结果。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
# This is our decorator
def
simple_decorator
(
f
)
:
# This is the new function we're going to return
# This function will be used in place of our original definition
def
wrapper
(
)
:
print
"Entering Function"
f
(
)
print
"Exited Function"
return
wrapper
@
simple_decorator
def
hello
(
)
:
print
"Hello World"
hello
(
|
还不是很明白?这里是一个简单的例子:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
# This is our decorator
def
simple_decorator
(
f
)
:
# This is the new function we're going to return
# This function will be used in place of our original definition
def
wrapper
(
)
:
print
"Entering Function"
f
(
)
print
"Exited Function"
return
wrapper
@
simple_decorator
def
hello
(
)
:
print
"Hello World"
hello
(
)
|
运行上述代码会输出以下结果:
Entering Function
Hello World
Exited Function
很好!
现在我们有点明白怎样创建我们自己的“@app.route()”装饰器了,但你可能会注意到有一个不同点,就是我们的simple_decorator不可以接受任何参数, 但“@app.route()”却可以。
那么我们怎样才能给我们的装饰器传参数?要实现这个我们只需创建一个“decorator_factory”函数,我们调用这个函数,返回适用于我们函数的装饰器。现在看看如果实现它。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
def
decorator_factory
(
enter_message
,
exit_message
)
:
# We're going to return this decorator
def
simple_decorator
(
f
)
:
def
wrapper
(
)
:
print
enter
_message
f
(
)
print
exit_message
return
wrapper
return
simple
_decorator
@
decorator_factory
(
"Start"
,
"End"
)
def
hello
(
)
:
print
"Hello World"
hello
(
)
|
给我们的输出是:
请注意在我们写@decorator_factory(“Start”, “End”)时,我们实际调用的是decorator_factory函数,实际返回的装饰器已经被用上了,代码很整洁,对吧?
Start
Hello World
End
把“app”放进“app.route”
现在我们掌握了装饰器怎样工作的全部前置知识 ,可以重新实现Flask API的这个部分了,那么把我们的目光转移到“app”在我们Flask应用中的重要地位上面来。
在开始解释Flask对象里面发生了什么之前,我们先创建我们自己的Python类NotFlask。
1
2
3
4
|
class
NotFlask
(
)
:
pass
app
=
NotFlask
(
)
|
这不是个很有趣的类,不过有一样值得注意,就是这个类的方法也可以被用作装饰器,所以让我们把这个类写得更有趣一点,加一个称作 route的方法,它是一个简单的装饰器工厂。
1
2
3
4
5
6
7
8
9
10
11
12
|
class
NotFlask
(
)
:
def
route
(
self
,
route_str
)
:
def
decorator
(
f
)
:
return
f
return
decorator
app
=
NotFlask
(
)
@
app
.
route
(
"/"
)
def
hello
(
)
:
return
"Hello World!"
|
这个装饰器和我们之前创建的那些最大的不同,在于我们不想修改被我们装饰的函数的行为,我们只是想获得它的引用。
所以,最后一步是我们打算去利用一个特性,就是用装饰器函数的副产品去保存一个提供给我们的路径之间的链接,装饰器函数应该与它关联起来。
为了实现这个,我们给我们的NotFlask对象加一个“routes”字典,当我们的“decorator”函数被调用,路径将被插入新字典中函数对应的位置。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
class
NotFlask
(
)
:
def
__init__
(
self
)
:
self
.
routes
=
{
}
def
route
(
self
,
route_str
)
:
def
decorator
(
f
)
:
self
.
routes
[
route_str
]
=
f
return
f
return
decorator
app
=
NotFlask
(
)
@
app
.
route
(
"/"
)
def
hello
(
)
:
return
"Hello World!"
|
现在我们就要完成了!可如果没法访问内部的视图函数,保存路径的字典又有什么用?让我们加入一个方法serve(path),当给定的路径存在时运行一个函数并给们我结果,当路径尚未注册时则抛出一个异常。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
|
class
NotFlask
(
)
:
def
__init__
(
self
)
:
self
.
routes
=
{
}
def
route
(
self
,
route_str
)
:
def
decorator
(
f
)
:
self
.
routes
[
route_str
]
=
f
return
f
return
decorator
def
serve
(
self
,
path
)
:
view_function
=
self
.
routes
.
get
(
path
)
if
view_function
:
return
view_function
(
)
else
:
raise
ValueError
(
'Route "{}"" has not been registered'
.
format
(
path
)
)
app
=
NotFlask
(
)
@
app
.
route
(
"/"
)
def
hello
(
)
:
return
"Hello World!"
|
在这个系列我们只关注重现那些热门库提供的友好API,所以钩挂“serve”方法实现一个HTTP服务器其实有一点超出本文的范围,当然结果是确定的,运行下述片段:
1
2
3
4
5
6
7
|
app
=
NotFlask
(
)
@
app
.
route
(
"/"
)
def
hello
(
)
:
return
"Hello World!"
print
app
.
serve
(
"/"
)
|
我们会看到:
Hello World!
我们已经完成了一个的Flask网页上第一个例子的非常简单的重现,让我们写一些快速测试检测我们简单重现的Flask的“@app.route()”是否正确。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
class
TestNotFlask
(
unittest
.
TestCase
)
:
def
setUp
(
self
)
:
self
.
app
=
NotFlask
(
)
def
test_valid_route
(
self
)
:
@
self
.
app
.
route
(
'/'
)
def
index
(
)
:
return
'Hello World'
self
.
assertEqual
(
self
.
app
.
serve
(
'/'
)
,
'Hello World'
)
def
test_invalid_route
(
self
)
:
with
self
.
assertRaises
(
ValueError
)
:
self
.
app
.
serve
(
'/invalid'
)
|
吸口气。
完全正确!所以,仅仅是一个简单的包含一个字典的装饰器, 就重现了Flask的“app.route()”装饰器的基本的行为。
在本系列的下一篇,也是Flask的app.route()的最后一篇,将通过解析下面这个例子来解释动态URL模式是如何工作。
1
2
3
4
5
|
;
html
-
script
:
false
]
app
=
Flask
(
__name__
)
@
app
.
route
(
"/hello/<username>"
)
def
hello_user
(
username
)
:
return
"Hello {} !"
.
format
(
username
)
|
待续!