成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

flask 源碼解析:路由

freecode / 3414人閱讀

摘要:而對(duì)于動(dòng)態(tài)路由,還需要更復(fù)雜的匹配邏輯。路由邏輯事實(shí)上,核心的路由邏輯是在中實(shí)現(xiàn)的。所以在繼續(xù)分析之前,我們先看一下提供的路由功能??梢园l(fā)現(xiàn),在路由過程中非常重要。的路由過程,其實(shí)是到的轉(zhuǎn)換通過找到處理該的。

文章屬于作者原創(chuàng),原文發(fā)布在個(gè)人博客。

構(gòu)建路由規(guī)則

一個(gè) web 應(yīng)用不同的路徑會(huì)有不同的處理函數(shù),路由就是根據(jù)請(qǐng)求的 URL 找到對(duì)應(yīng)處理函數(shù)的過程。

在執(zhí)行查找之前,需要有一個(gè)規(guī)則列表,它存儲(chǔ)了 url 和處理函數(shù)的對(duì)應(yīng)關(guān)系。最容易想到的解決方案就是定義一個(gè)字典,key 是 url,value 是對(duì)應(yīng)的處理函數(shù)。如果 url 都是靜態(tài)的(url 路徑都是實(shí)現(xiàn)確定的,沒有變量和正則匹配),那么路由的過程就是從字典中通過 url 這個(gè) key ,找到并返回對(duì)應(yīng)的 value;如果沒有找到,就報(bào) 404 錯(cuò)誤。而對(duì)于動(dòng)態(tài)路由,還需要更復(fù)雜的匹配邏輯。flask 中的路由過程是這樣的嗎?這篇文章就來分析分析。

在分析路由匹配過程之前,我們先來看看 flask 中,構(gòu)建這個(gè)路由規(guī)則的兩種方法:

通過 @app.route() decorator,比如文章開頭給出的 hello world 例子

通過 app.add_url_rule,這個(gè)方法的簽名為 add_url_rule(self, rule, endpoint=None, view_func=None, **options),參數(shù)的含義如下:

rule: url 規(guī)則字符串,可以是靜態(tài)的 /path,也可以包含 /

endpoint:要注冊(cè)規(guī)則的 endpoint,默認(rèn)是 view_func 的名字

view_func:對(duì)應(yīng) url 的處理函數(shù),也被稱為視圖函數(shù)

這兩種方法是等價(jià)的,也就是說:

@app.route("/")
def hello():
    return "hello, world!"

也可以寫成

def hello():
    return "hello, world!"

app.add_url_rule("/", "hello", hello)

NOTE: 其實(shí),還有一種方法來構(gòu)建路由規(guī)則——直接操作 app.url_map 這個(gè)數(shù)據(jù)結(jié)構(gòu)。不過這種方法并不是很常用,因此就不展開了。

注冊(cè)路由規(guī)則的時(shí)候,flask 內(nèi)部做了哪些東西呢?我們來看看 route 方法:

def route(self, rule, **options):
    """A decorator that is used to register a view function for a
    given URL rule.  This does the same thing as :meth:`add_url_rule`
    but is intended for decorator usage.
    """

    def decorator(f):
        endpoint = options.pop("endpoint", None)
        self.add_url_rule(rule, endpoint, f, **options)
        return f

    return decorator

route 方法內(nèi)部也是調(diào)用 add_url_rule,只不過在外面包了一層裝飾器的邏輯,這也驗(yàn)證了上面兩種方法等價(jià)的說法。

def add_url_rule(self, rule, endpoint=None, view_func=None, **options):
    """Connects a URL rule.  Works exactly like the :meth:`route`
    decorator.  If a view_func is provided it will be registered with the
    endpoint.
    """

    methods = options.pop("methods", None)

    rule = self.url_rule_class(rule, methods=methods, **options)
    self.url_map.add(rule)
    if view_func is not None:
        old_func = self.view_functions.get(endpoint)
        if old_func is not None and old_func != view_func:
            raise AssertionError("View function mapping is overwriting an "
                                 "existing endpoint function: %s" % endpoint)
        self.view_functions[endpoint] = view_func

上面這段代碼省略了處理 endpoint 和構(gòu)建 methods 的部分邏輯,可以看到它主要做的事情就是更新 self.url_mapself.view_functions 兩個(gè)變量。找到變量的定義,發(fā)現(xiàn) url_mapwerkzeug.routeing:Map 類的對(duì)象,rulewerkzeug.routing:Rule 類的對(duì)象,view_functions 就是一個(gè)字典。這和我們之前預(yù)想的并不一樣,這里增加了 RuleMap 的封裝,還把 urlview_func 保存到了不同的地方。

需要注意的是:每個(gè)視圖函數(shù)的 endpoint 必須是不同的,否則會(huì)報(bào) AssertionError

werkzeug 路由邏輯

事實(shí)上,flask 核心的路由邏輯是在 werkzeug 中實(shí)現(xiàn)的。所以在繼續(xù)分析之前,我們先看一下 werkzeug 提供的路由功能。

>>> m = Map([
...     Rule("/", endpoint="index"),
...     Rule("/downloads/", endpoint="downloads/index"),
...     Rule("/downloads/", endpoint="downloads/show")
... ])
>>> urls = m.bind("example.com", "/")
>>> urls.match("/", "GET")
("index", {})
>>> urls.match("/downloads/42")
("downloads/show", {"id": 42})

>>> urls.match("/downloads")
Traceback (most recent call last):
  ...
RequestRedirect: http://example.com/downloads/
>>> urls.match("/missing")
Traceback (most recent call last):
  ...
NotFound: 404 Not Found

上面的代碼演示了 werkzeug 最核心的路由功能:添加路由規(guī)則(也可以使用 m.add),把路由表綁定到特定的環(huán)境(m.bind),匹配url(urls.match)。正常情況下返回對(duì)應(yīng)的 endpoint 名字和參數(shù)字典,可能報(bào)重定向或者 404 異常。

可以發(fā)現(xiàn),endpoint 在路由過程中非常重要。werkzeug 的路由過程,其實(shí)是 url 到 endpoint 的轉(zhuǎn)換:通過 url 找到處理該 url 的 endpoint。至于 endpoint 和 view function 之間的匹配關(guān)系,werkzeug 是不管的,而上面也看到 flask 是把這個(gè)存放到字典中的。

flask 路由實(shí)現(xiàn)

好,有了這些基礎(chǔ)知識(shí),我們回頭看 dispatch_request,繼續(xù)探尋路由匹配的邏輯:

def dispatch_request(self):
    """Does the request dispatching.  Matches the URL and returns the
    return value of the view or error handler.  This does not have to
    be a response object.  In order to convert the return value to a
    proper response object, call :func:`make_response`.
    """

    req = _request_ctx_stack.top.request
    if req.routing_exception is not None:
        self.raise_routing_exception(req)
    rule = req.url_rule

    # dispatch to the handler for that endpoint
    return self.view_functions[rule.endpoint](**req.view_args)

這個(gè)方法做的事情就是找到請(qǐng)求對(duì)象 request,獲取它的 endpoint,然后從 view_functions 找到對(duì)應(yīng) endpointview_func ,把請(qǐng)求參數(shù)傳遞過去,進(jìn)行處理并返回。view_functions 中的內(nèi)容,我們已經(jīng)看到,是在構(gòu)建路由規(guī)則的時(shí)候保存進(jìn)去的;那請(qǐng)求中 req.url_rule 是什么保存進(jìn)去的呢?它的格式又是什么?

我們可以先這樣理解:_request_ctx_stack.top.request 保存著當(dāng)前請(qǐng)求的信息,在每次請(qǐng)求過來的時(shí)候,flask 會(huì)把當(dāng)前請(qǐng)求的信息保存進(jìn)去,這樣我們就能在整個(gè)請(qǐng)求處理過程中使用它。至于怎么做到并發(fā)情況下信息不會(huì)相互干擾錯(cuò)亂,我們將在下一篇文章介紹。

_request_ctx_stack 中保存的是 RequestContext 對(duì)象,它出現(xiàn)在 flask/globals.py 文件中,和路由相關(guān)的邏輯如下:

class RequestContext(object):
    def __init__(self, app, environ, request=None):
        self.app = app
        self.request = request
        self.url_adapter = app.create_url_adapter(self.request)
        self.match_request()
    
    def match_request(self):
        """Can be overridden by a subclass to hook into the matching
        of the request.
        """
        try:
            url_rule, self.request.view_args = 
                self.url_adapter.match(return_rule=True)
            self.request.url_rule = url_rule
        except HTTPException as e:
            self.request.routing_exception = e

            
class Flask(_PackageBoundObject):
    def create_url_adapter(self, request):
        """Creates a URL adapter for the given request.  The URL adapter
        is created at a point where the request context is not yet set up
        so the request is passed explicitly.
        """
        if request is not None:
            return self.url_map.bind_to_environ(request.environ,
                server_name=self.config["SERVER_NAME"])

在初始化的時(shí)候,會(huì)調(diào)用 app.create_url_adapter 方法,把 appurl_map 綁定到 WSGI environ 變量上(bind_to_environ 和之前的 bind 方法作用相同)。最后會(huì)調(diào)用 match_request 方法,這個(gè)方式調(diào)用了 url_adapter.match 方法,進(jìn)行實(shí)際的匹配工作,返回匹配的 url rule。而我們之前使用的 url_rule.endpoint 就是匹配的 endpoint 值。

整個(gè) flask 的路由過程就結(jié)束了,總結(jié)一下大致的流程:

通過 @app.route 或者 app.add_url_rule 注冊(cè)應(yīng)用 url 對(duì)應(yīng)的處理函數(shù)

每次請(qǐng)求過來的時(shí)候,會(huì)事先調(diào)用路由匹配的邏輯,把路由結(jié)果保存起來

dispatch_request 根據(jù)保存的路由結(jié)果,調(diào)用對(duì)應(yīng)的視圖函數(shù)

match 實(shí)現(xiàn)

雖然講完了 flask 的路由流程,但是還沒有講到最核心的問題:werkzeug 中是怎么實(shí)現(xiàn) match 方法的。Map 保存了 Rule 列表,match 的時(shí)候會(huì)依次調(diào)用其中的 rule.match 方法,如果匹配就找到了 match。Rule.match 方法的代碼如下:

def match(self, path):
        """Check if the rule matches a given path. Path is a string in the
        form ``"subdomain|/path(method)"`` and is assembled by the map.  If
        the map is doing host matching the subdomain part will be the host
        instead.

        If the rule matches a dict with the converted values is returned,
        otherwise the return value is `None`.
        """
        if not self.build_only:
            m = self._regex.search(path)
            if m is not None:
                groups = m.groupdict()

                result = {}
                for name, value in iteritems(groups):
                    try:
                        value = self._converters[name].to_python(value)
                    except ValidationError:
                        return
                    result[str(name)] = value
                if self.defaults:
                    result.update(self.defaults)

                return result

它的邏輯是這樣的:用實(shí)現(xiàn) compile 的正則表達(dá)式去匹配給出的真實(shí)路徑信息,把所有的匹配組件轉(zhuǎn)換成對(duì)應(yīng)的值,保存在字典中(這就是傳遞給視圖函數(shù)的參數(shù)列表)并返回。

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/38371.html

相關(guān)文章

  • flask 源碼解析:簡(jiǎn)介

    摘要:簡(jiǎn)介官網(wǎng)上對(duì)它的定位是一個(gè)微開發(fā)框架。另外一個(gè)必須理解的概念是,簡(jiǎn)單來說就是一套和框架應(yīng)用之間的協(xié)議。功能比較豐富,支持解析自動(dòng)防止攻擊繼承變量過濾器流程邏輯支持代碼邏輯集成等等。那么,從下一篇文章,我們就正式開始源碼之旅了 文章屬于作者原創(chuàng),原文發(fā)布在個(gè)人博客。 flask 簡(jiǎn)介 Flask 官網(wǎng)上對(duì)它的定位是一個(gè)微 python web 開發(fā)框架。 Flask is a micro...

    megatron 評(píng)論0 收藏0
  • flask 源碼解析:響應(yīng)

    摘要:我們知道響應(yīng)分為三個(gè)部分狀態(tài)欄版本狀態(tài)碼和說明頭部以冒號(hào)隔開的字符對(duì),用于各種控制和協(xié)商服務(wù)端返回的數(shù)據(jù)。 這是 flask 源碼解析系列文章的其中一篇,本系列所有文章列表: flask 源碼解析:簡(jiǎn)介 flask 源碼解析:應(yīng)用啟動(dòng)流程 flask 源碼解析:路由 flask 源碼解析:上下文 flask 源碼解析:請(qǐng)求 flask 源碼解析:響應(yīng) response 簡(jiǎn)介 在 f...

    wslongchen 評(píng)論0 收藏0
  • flask 源碼解析:應(yīng)用啟動(dòng)流程

    摘要:中有一個(gè)非常重要的概念每個(gè)應(yīng)用都是一個(gè)可調(diào)用的對(duì)象。它規(guī)定了的接口,會(huì)調(diào)用,并傳給它兩個(gè)參數(shù)包含了請(qǐng)求的所有信息,是處理完之后需要調(diào)用的函數(shù),參數(shù)是狀態(tài)碼響應(yīng)頭部還有錯(cuò)誤信息。一般來說,嵌套的最后一層是業(yè)務(wù)應(yīng)用,中間就是。 文章屬于作者原創(chuàng),原文發(fā)布在個(gè)人博客。 WSGI 所有的 python web 框架都要遵循 WSGI 協(xié)議,如果對(duì) WSGI 不清楚,可以查看我之前的介紹文章。 ...

    whatsns 評(píng)論0 收藏0
  • flask源碼走讀

    摘要:另外,如果你對(duì)模板渲染部分的內(nèi)容感興趣,也可以考慮閱讀文檔文檔文檔源碼閱讀,可以參考下面的函數(shù)打斷點(diǎn),再測(cè)試一個(gè)請(qǐng)求,理清過程。 Flask-Origin 源碼版本 一直想好好理一下flask的實(shí)現(xiàn),這個(gè)項(xiàng)目有Flask 0.1版本源碼并加了注解,挺清晰明了的,我在其基礎(chǔ)上完成了對(duì)Werkzeug的理解部分,大家如果想深入學(xué)習(xí)的話,可以參考werkzeug_flow.md. 閱讀前 為...

    Coly 評(píng)論0 收藏0
  • flask 源碼解析:請(qǐng)求

    摘要:可以看到,雖然是同樣的請(qǐng)求數(shù)據(jù),在不同的階段和不同組件看來,是完全不同的形式。請(qǐng)求還有一個(gè)不那么明顯的特性它不能被應(yīng)用修改,應(yīng)用只能讀取請(qǐng)求的數(shù)據(jù)。 這是 flask 源碼解析系列文章的其中一篇,本系列所有文章列表: flask 源碼解析:簡(jiǎn)介 flask 源碼解析:應(yīng)用啟動(dòng)流程 flask 源碼解析:路由 flask 源碼解析:上下文 flask 源碼解析:請(qǐng)求 flask 源碼解...

    weizx 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<