前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >Python Web Flask源码解读(一)——启动流程

Python Web Flask源码解读(一)——启动流程

作者头像
阳仔
发布于 2019-07-30 06:52:13
发布于 2019-07-30 06:52:13
94600
代码可运行
举报
文章被收录于专栏:终身开发者终身开发者
运行总次数:0
代码可运行

关于我 编程界的一名小小程序猿,目前在一个创业团队任team lead,技术栈涉及AndroidPythonJava和Go,这个也是我们团队的主要技术栈。 联系:hylinux1024@gmail.com 微信公众号:angrycode

0x00 什么是WSGI

WebServerGatewayInterface 它由 Python标准定义的一套 WebServerWebApplication接口交互规范

WSGI不是一个应用、框架、模块或者库,而是规范。

那什么是 WebServerWeb服务器)和什么是 WebApplicationWeb 应用)呢? 举例子来说明容易理解,例如常见的 Web应用框架有 DjangoFlask等,而 Web服务器有 uWSGIGunicorn等。WSGI就是定义了这两端接口交互的规范。

0x01 什么是Werkzeug

Werkzeug is a comprehensive WSGI web application library.

Werkzeug是一套实现 WSGI规范的函数库。我们可以使用它来创建一个 WebApplicationWeb应用)。例如本文介绍的 Flask应用框架就是基于 Werkzeug来开发的。

这里我们使用 Werkzeug启动一个简单的服务器应用

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
from werkzeug.wrappers import Request, Response

@Request.application
def application(request):
    return Response('Hello, World!')

if __name__ == '__main__':
    from werkzeug.serving import run_simple

    run_simple('localhost', 4000, application)

运行之后可以在控制台上将看到如下信息

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
* Running on http://localhost:4000/ (Press CTRL+C to quit)

使用浏览器打开 http://localhost:4000/ 看到以下信息,说明

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
Hello, World!

0x02 什么是Flask

Flask is a lightweight WSGI web application framework.

Flask是一个轻量级的 web应用框架,它是跑在 web服务器中的一个应用。Flask底层就是封装的 Werkzeug

使用 Flask开发一个 web应用非常简单

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
from flask import Flask

app = Flask(__name__)

@app.route('/')
def hello():
    return f'Hello, World!'

if __name__ == '__main__':
    app.run()

很简单吧。

接下来我们看看 Flask应用的启动流程。

0x03 启动流程

从项目地址 https://github.com/pallets/flask 中把源码 clone下来,然后切换到0.1版本的 tag。为何要使用0.1版本呢?因为这个是作者最开始写的版本,代码量应该是最少的,而且可以很容易看到作者整体编码思路。

下面就从最简单的 Demo开始看看 Flask是如何启动的。我们知道程序启动是执行了以下方法

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
if __name__ == '__main__':
    app.run()

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
app = Flask(__name__)

打开 Flask源码中的 __init__方法

Flask.init()
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
def __init__(self, package_name):
        #: 是否打开debug模式
        self.debug = False

        #: 包名或模块名
        self.package_name = package_name

        #: 获取app所在目录
        self.root_path = _get_package_path(self.package_name)

        #: 存储视图函数的字典,键为函数名称,值为函数对象,使用@route装饰器进行注册
        self.view_functions = {}

        #: 存储错误处理的字典.  键为error code, 值为处理错误的函数,使用errorhandler装饰器进行注册
        self.error_handlers = {}

        #: 处理请求前执行的函数列表,使用before_request装饰器进行注册
        self.before_request_funcs = []

        #: 处理请求前执行的函数列表,使用after_request装饰器进行注册
        self.after_request_funcs = []

        #: 模版上下文
        self.template_context_processors = [_default_template_ctx_processor]
        #: url 映射
        self.url_map = Map()

        #: 静态文件
        if self.static_path is not None:
            self.url_map.add(Rule(self.static_path + '/<filename>',
                                  build_only=True, endpoint='static'))
            if pkg_resources is not None:
                target = (self.package_name, 'static')
            else:
                target = os.path.join(self.root_path, 'static')
            self.wsgi_app = SharedDataMiddleware(self.wsgi_app, {
                self.static_path: target
            })

        #: 初始化 Jinja2 模版环境.
        self.jinja_env = Environment(loader=self.create_jinja_loader(),
                                     **self.jinja_options)
        self.jinja_env.globals.update(
            url_for=url_for,
            get_flashed_messages=get_flashed_messages
        )

Flask的构造函数中进行了各种初始化操作。

然后就是执行 app.run()方法

app.run()
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
def run(self, host='localhost', port=5000, **options):
    """启动本地开发服务器.  如果debug设置为True,那么会自动检查代码是否改动,有改动则会自动执行部署
    :param host: 监听的IP地址. 如果设置为 ``'0.0.0.0'``就可以进行外部访问
    :param port: 端口,默认5000
    :param options: 这个参数主要是对应run_simple中需要的参数
    """
    from werkzeug.serving import run_simple
    if 'debug' in options:
        self.debug = options.pop('debug')
    options.setdefault('use_reloader', self.debug)
    options.setdefault('use_debugger', self.debug)
    return run_simple(host, port, self, **options)

run很简洁,主要是调用了 werkzeug.serving中的 run_simple方法。

再打开 run_simple的源码

rum_simple()
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
def run_simple(hostname, port, application, use_reloader=False,
               use_debugger=False, use_evalex=True,
               extra_files=None, reloader_interval=1, threaded=False,
               processes=1, request_handler=None, static_files=None,
               passthrough_errors=False, ssl_context=None):
    # 这方法还是比较短的,但是注释写得很详细,由于篇幅问题,就把源码中的注释省略了
    if use_debugger:
        from werkzeug.debug import DebuggedApplication
        application = DebuggedApplication(application, use_evalex)
    if static_files:
        from werkzeug.wsgi import SharedDataMiddleware
        application = SharedDataMiddleware(application, static_files)

    def inner():
        make_server(hostname, port, application, threaded,
                    processes, request_handler,
                    passthrough_errors, ssl_context).serve_forever()

    if os.environ.get('WERKZEUG_RUN_MAIN') != 'true':
        display_hostname = hostname != '*' and hostname or 'localhost'
        if ':' in display_hostname:
            display_hostname = '[%s]' % display_hostname
        _log('info', ' * Running on %s://%s:%d/', ssl_context is None
             and 'http' or 'https', display_hostname, port)
    if use_reloader:
        # Create and destroy a socket so that any exceptions are raised before
        # we spawn a separate Python interpreter and lose this ability.
        test_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
        test_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
        test_socket.bind((hostname, port))
        test_socket.close()
        run_with_reloader(inner, extra_files, reloader_interval)
    else:
        inner()

rum_simple方法中还定义一个嵌套方法 inner(),这个是方法的核心部分。

inner()
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
def inner():
    make_server(hostname, port, application, threaded, processes, request_handler, passthrough_errors, ssl_context).serve_forever()

inner()方法里面,调用 make_server(...).serve_forever()启动了服务。

make_server()
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
def make_server(host, port, app=None, threaded=False, processes=1,
                request_handler=None, passthrough_errors=False,
                ssl_context=None):
    """Create a new server instance that is either threaded, or forks
    or just processes one request after another.
    """
    if threaded and processes > 1:
        raise ValueError("cannot have a multithreaded and "
                         "multi process server.")
    elif threaded:
        return ThreadedWSGIServer(host, port, app, request_handler,
                                  passthrough_errors, ssl_context)
    elif processes > 1:
        return ForkingWSGIServer(host, port, app, processes, request_handler,
                                 passthrough_errors, ssl_context)
    else:
        return BaseWSGIServer(host, port, app, request_handler,
                              passthrough_errors, ssl_context)

make_server()中会根据线程或者进程的数量创建对应的 WSGI服务器。Flask在默认情况下是创建 BaseWSGIServer服务器。

BaseWSGIServer、ThreadedWSGIServer、ForkingWSGIServer
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
class BaseWSGIServer(HTTPServer, object):
    ...

class ThreadedWSGIServer(ThreadingMixIn, BaseWSGIServer):
    """A WSGI server that does threading."""
    ...

class ForkingWSGIServer(ForkingMixIn, BaseWSGIServer):
    """A WSGI server that does forking."""
    ...

可以看出他们之前的继承关系如下

打开 BaseWSGIServerstart_server()方法

start_server()
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
def serve_forever(self):
    try:
        HTTPServer.serve_forever(self)
    except KeyboardInterrupt:
        pass

可以看到最终是使用 HTTPServer中的启动服务的方法。而 HTTPServerPython标准类库中的接口。

HTTPServersocketserver.TCPServer的子类

socketserver.TCPServer

如果要使用 Python中类库启动一个 http server,则类似代码应该是这样的

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
import http.server
import socketserver

PORT = 8000

Handler = http.server.SimpleHTTPRequestHandler

with socketserver.TCPServer(("", PORT), Handler) as httpd:
    print("serving at port", PORT)
    httpd.serve_forever()

至此,整个服务的启动就到这里就启动起来了。

这个过程的调用流程为

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
graph TD

A[Flask]-->B[app.run]
B[app.run]-->C[werkzeug.run_simple]
C[werkzeug.run_simple]-->D[BaseWSGIServer]
D[BaseWSGIServer]-->E[HTTPServer.serve_forever]
E[HTTPServer.serve_forever]-->F[TCPServer.serve_forever]

0x04 总结一下

WSGIWEB服务器与 WEB应用之间交互的接口规范。werkzeug是实现了这一个规范的函数库,而 Flask框架是基于 werkzeug来实现的。我们从 Flask.run()方法启动服务开始,追踪了整个服务启动的流程。

0x05 学习资料

  • https://werkzeug.palletsprojects.com/en/0.15.x/
  • https://palletsprojects.com/p/flask/
  • https://docs.python.org/3/library/http.server.html#module-http.server
本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2019-07-23,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 终身开发者 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
7-3 打印沙漏
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
韩旭051
2019/11/08
4610
打印沙漏 C语言
本题要求你写个程序把给定的符号打印成沙漏的形状。例如给定17个“*”,要求按下列格式打印
叶茂林
2023/07/28
2830
1027 打印沙漏 (20 分)
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
韩旭051
2019/11/08
3690
C语言实例练习(上)
对某些题目做了一些小改动,并加入了自己的学习笔记和理解,代码不是原教程中的代码,是我自己作为练习写的,每块代码都测试了,应该是没有问题,但不足之处仍无可避免,如有问题,还请各位大佬批评指正
小孙同学
2022/01/14
3.2K0
PTA刷题记录:L1-002 打印沙漏 (20分)
题目要求: 本题要求你写个程序把给定的符号打印成沙漏的形状。例如给定17个“*”,要求按下列格式打印
英雄爱吃土豆片
2020/10/29
1.6K0
PTA刷题记录:L1-002 打印沙漏 (20分)
C:图案打印
3.你可以将图案看作在一张表格上,对于每个位置( i , j ),通过条件判断 i == j 表示主对角线(从左上角到右下角)上的位置,
LonlyMay
2024/10/21
1350
C:图案打印
c语言每日一练(2)
A.(2)=(3) B.(1)=(2) C.都不一样 D.都一样
大海里的番茄
2024/01/19
2120
c语言每日一练(2)
打印菱形Java_for循环打印菱形
1、星号前面的空白要用空格代替。 2、把图形分为上下两部分,分别找出行数与“空格”和“*”的关系
全栈程序员站长
2022/11/11
3.1K0
打印菱形Java_for循环打印菱形
c语言中的常见图形打印
通过观察,不难发现,每一行的元素的规律是2×i-1;(i表示行数) 所以我们的代码可以这样写:
初阶牛
2023/02/10
2K0
c语言中的常见图形打印
C语言实例:创建各类三角形图案(杨辉三角,弗洛伊德三角形....)
红框里的代码很重要,没有这句话,三角形就打印不出来,打印的只是许多连起来的‘*’。
aosei
2024/01/23
2350
C语言实例:创建各类三角形图案(杨辉三角,弗洛伊德三角形....)
C语言之——入门必刷题(2)
在一些入门题目中,题目经常会让我们输入n个数,之后以升序或者降序的方式输出,或者比较。
The sky
2023/04/12
5250
C语言之——入门必刷题(2)
如何使用C语言打印三角形和菱形?
通过观察,不难发现,每一行的元素的规律是2×i-1;(i表示行数) 所以我们的代码可以这样写:
初阶牛
2023/10/14
3630
如何使用C语言打印三角形和菱形?
案列:流程控制练习案例
一、中奖的概率 判断一个数需要随机多少次才能中奖,打印随机次数 import random num = 432 # for i in range(100,500): i = 0 while True: Winning = random.randrange(100,500) #产生一个区间范围的随机数 i+=1 if num == Winning: print("中奖了 中奖号码是{}".format(num)) break print(i) 二、求一
星哥玩云
2022/09/08
4300
Java|实现图形打印
在Java学习当中,当我们熟练掌握了关于for循环的基础之后,就可以利用其来实现一个图形的打印,主要是利用“*”符号或者其他符号来进行图形的一个拼合,来呈现出一个完整图形样式。
算法与编程之美
2020/08/06
2.6K0
用for循环语句实现在屏幕上打印特殊图案编程题目的解法
(题目来源于牛客网题库)链接:https://www.nowcoder.com/ta/beginner-programmers
小孙同学
2022/01/14
2.3K0
【蓝桥杯Java_C组·从零开始卷】第三节(附)、for循环练习题(数据题与图形题)
*  *  *  *  *  *  *  *  *  *  *  *  *  *  * *  *  *  *  *  *  *  *  *  * 
红目香薰
2022/11/29
3710
【蓝桥杯Java_C组·从零开始卷】第三节(附)、for循环练习题(数据题与图形题)
Java 实例-打印图形
我们可以看到,图形共5行,那么,我们是否可以建立一个for循环语句,使其控制在5行?答案是肯定的。
默 语
2024/11/20
980
Java 实例-打印图形
【重生之学习C语言----杨辉三角篇】
杨辉三角(Pascal's Triangle)是二项式系数在三角形中的一种几何排列。它具有以下特点:
用户11456817
2025/02/07
1450
python练习题参考答案来啦(2)
昨天放了第三篇的参考答案,仅供参考,想要学的更深入一些可以自己看一些算法类的书籍或者文章,应该会更系统和专业。
叶子陪你玩
2021/12/28
8770
python练习题参考答案来啦(2)
【Java案例】打印杨辉三角
图1.10 杨辉三角形 案例分析 观察杨辉三角形的图案,可以发现其中的规律:三角形的竖边和斜边都是“1”,三角形里面的任意一个数字正好等于它正上方的数字和左上角的数字两个数字之和。第几行就有几个数字
Java帮帮
2018/03/15
2.5K0
【Java案例】打印杨辉三角
相关推荐
7-3 打印沙漏
更多 >
LV.1
这个人很懒,什么都没有留下~
目录
  • 0x00 什么是WSGI
  • 0x01 什么是Werkzeug
  • 0x02 什么是Flask
  • 0x03 启动流程
    • Flask.init()
    • app.run()
    • rum_simple()
    • inner()
    • make_server()
      • BaseWSGIServer、ThreadedWSGIServer、ForkingWSGIServer
      • start_server()
      • socketserver.TCPServer
  • 0x04 总结一下
  • 0x05 学习资料
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档