欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Django應(yīng)用程序入口WSGIHandler源碼解析

 更新時(shí)間:2019年08月05日 11:12:40   作者:搗亂小子  
這篇文章主要介紹了Django應(yīng)用程序入口WSGIHandler源碼解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下

前言

WSGI 有三個(gè)部分, 分別為服務(wù)器(server), 應(yīng)用程序(application) 和中間件(middleware). 已經(jīng)知道, 服務(wù)器方面會調(diào)用應(yīng)用程序來處理請求, 在應(yīng)用程序中有真正的處理邏輯, 在這里面幾乎可以做任何事情, 其中的中間件就會在里面展開.

Django 中的應(yīng)用程序

任何的 WSGI 應(yīng)用程序, 都必須是一個(gè) start_response(status, response_headers, exc_info=None) 形式的函數(shù)或者定義了 __call__ 的類. 而 django.core.handlers 就用后一種方式實(shí)現(xiàn)了應(yīng)用程序: WSGIHandler. 在這之前, Django 是如何指定自己的 application 的, 在一個(gè)具體的 Django 項(xiàng)目中, 它的方式如下:

在 mysite.settings.py 中能找到如下設(shè)置:

# Python dotted path to the WSGI application used by Django's runserver.
WSGI_APPLICATION = 'tomato.wsgi.application'

如你所見, WSGI_APPLICATION 就指定了應(yīng)用程序. 而按圖索驥下去, 找到項(xiàng)目中的 wsgi.py, 已經(jīng)除去了所有的注釋:

import os 
os.environ.setdefault("DJANGO_SETTINGS_MODULE", "tomato.settings")
from django.core.wsgi import get_wsgi_application
application = get_wsgi_application()

因此, WSGI_APPLICATION 所指定的即為 wsgi.py 中的全局變量 application. 故伎重演, 繼續(xù)找下去. 在 django.core 模塊中的 wsgi.py 中找到 get_wsgi_application() 函數(shù)的實(shí)現(xiàn):

from django.core.handlers.wsgi import WSGIHandler
def get_wsgi_application():
  """
  The public interface to Django's WSGI support. Should return a WSGI
  callable. 
  Allows us to avoid making django.core.handlers.WSGIHandler public API, in
  case the internal WSGI implementation changes or moves in the future.
 
  """
  """
  # 繼承, 但只實(shí)現(xiàn)了 __call__ 方法, 方便使用
  class WSGIHandler(base.BaseHandler):
  """
  return WSGIHandler()

在 get_wsgi_application() 中實(shí)例化了 WSGIHandler, 并無其他操作.

WSGIHandler

緊接著在 django.core.handler 的 base.py 中找到 WSGIHandler 的實(shí)現(xiàn).

# 繼承, 但只實(shí)現(xiàn)了 __call__ 方法, 方便使用
class WSGIHandler(base.BaseHandler):
  initLock = Lock() 
  # 關(guān)于此, 日后展開, 可以將其視為一個(gè)代表 http 請求的類
  request_class = WSGIRequest 
  # WSGIHandler 也可以作為函數(shù)來調(diào)用
  def __call__(self, environ, start_response):
    # Set up middleware if needed. We couldn't do this earlier, because
    # settings weren't available. 
    # 這里的檢測: 因?yàn)?self._request_middleware 是最后才設(shè)定的, 所以如果為空,
    # 很可能是因?yàn)?self.load_middleware() 沒有調(diào)用成功.
    if self._request_middleware is None:
      with self.initLock:
        try:
          # Check that middleware is still uninitialised.
          if self._request_middleware is None:
            因?yàn)?load_middleware() 可能沒有調(diào)用, 調(diào)用一次.
            self.load_middleware()
        except:
          # Unload whatever middleware we got
          self._request_middleware = None
          raise 
    set_script_prefix(base.get_script_name(environ))
    signls.request_started.send(sender=self.__class__) # __class__ 代表自己的類 
    try:
      # 實(shí)例化 request_class = WSGIRequest, 將在日后文章中展開, 可以將其視為一個(gè)代表 http 請求的類
      request = self.request_class(environ)
 
    except UnicodeDecodeError:
      logger.warning('Bad Request (UnicodeDecodeError)',
        exc_info=sys.exc_info(),
        extra={
          'status_code': 400,
        }
      )
      response = http.HttpResponseBadRequest()
    else:
      # 調(diào)用 self.get_response(), 將會返回一個(gè)相應(yīng)對象 response<br>      ############# 關(guān)鍵的操作, self.response() 可以獲取響應(yīng)數(shù)據(jù).     
      response = self.get_response(request)
 
    # 將 self 掛鉤到 response 對象
    response._handler_class = self.__class__ 
    try:
      status_text = STATUS_CODE_TEXT[response.status_code]
    except KeyError:
      status_text = 'UNKNOWN STATUS CODE'
     # 狀態(tài)碼
    status = '%s %s' % (response.status_code, status_text) 
    response_headers = [(str(k), str(v)) for k, v in response.items()] 
    # 對于每個(gè)一個(gè) cookie, 都在 header 中設(shè)置: Set-cookie xxx=yyy
    for c in response.cookies.values():
      response_headers.append((str('Set-Cookie'), str(c.output(header=''))))
 
    # start_response() 操作已經(jīng)在上節(jié)中介紹了
    start_response(force_str(status), response_headers) 
    # 成功返回相應(yīng)對象
    return response

WSGIHandler 類只實(shí)現(xiàn)了 def __call__(self, environ, start_response), 使它本身能夠成為 WSGI 中的應(yīng)用程序, 并且實(shí)現(xiàn) __call__ 能讓類的行為跟函數(shù)一樣, 詳見 python __call__ 方法.

def __call__(self, environ, start_response) 方法中調(diào)用了 WSGIHandler.get_response() 方法以獲取響應(yīng)數(shù)據(jù)對象 response. 從 WSGIHandler 的實(shí)現(xiàn)來看, 它并不是最為底層的: WSGIHandler 繼承自 base.BaseHandler, 在 django.core.handler 的 base.py 中可以找到: class BaseHandler(object):...

這一節(jié)服務(wù)器部分已經(jīng)結(jié)束, 接下來的便是中間件和應(yīng)用程序了, 相關(guān)內(nèi)容會在下節(jié)的 BaseHandler 中展開. 我已經(jīng)在 github 備份了 Django 源碼的注釋: Decode-Django, 有興趣的童鞋 fork 吧.

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • Python 線程池用法簡單示例

    Python 線程池用法簡單示例

    這篇文章主要介紹了Python 線程池用法,結(jié)合簡單實(shí)例形式分析了Python線程池相關(guān)使用技巧與操作注意事項(xiàng),需要的朋友可以參考下
    2019-10-10
  • Python Parser的用法

    Python Parser的用法

    這篇文章主要介紹了Python Parser的用法,文中有非常詳細(xì)的代碼示例,對正在學(xué)習(xí)python的小伙伴們有很好地幫助,需要的朋友可以參考下
    2021-05-05
  • 深度學(xué)習(xí)Tensorflow?2.4?完成遷移學(xué)習(xí)和模型微調(diào)

    深度學(xué)習(xí)Tensorflow?2.4?完成遷移學(xué)習(xí)和模型微調(diào)

    這篇文章主要為大家介紹了深度學(xué)習(xí)Tensorflow?2.4?完成遷移學(xué)習(xí)和模型微調(diào),有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-01-01
  • 一文詳解Python中生成器的原理與使用

    一文詳解Python中生成器的原理與使用

    生成器表達(dá)式本質(zhì)上就是一個(gè)迭代器,是定義迭代器的一種方式,是允許自定義邏輯的迭代器。本文將詳細(xì)講解一下Python中生成器的原理與使用,需要的可以參考一下
    2022-05-05
  • 解決django中form表單設(shè)置action后無法回到原頁面的問題

    解決django中form表單設(shè)置action后無法回到原頁面的問題

    這篇文章主要介紹了解決django中form表單設(shè)置action后無法回到原頁面的問題,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-03-03
  • python中ndarray數(shù)組的索引和切片的使用

    python中ndarray數(shù)組的索引和切片的使用

    本文主要介紹了python中ndarray數(shù)組的索引和切片的使用,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-07-07
  • Python實(shí)現(xiàn)遍歷子文件夾并將文件復(fù)制到不同的目標(biāo)文件夾

    Python實(shí)現(xiàn)遍歷子文件夾并將文件復(fù)制到不同的目標(biāo)文件夾

    這篇文章主要介紹了如何基于Python語言實(shí)現(xiàn)遍歷多個(gè)子文件夾,將每一個(gè)子文件夾中大量的文件,按照每一個(gè)文件的文件名稱的特點(diǎn)復(fù)制到不同的目標(biāo)文件夾中,感興趣的可以了解下
    2023-08-08
  • Python Socket編程詳細(xì)介紹

    Python Socket編程詳細(xì)介紹

    這篇文章主要介紹了Python Socket編程詳細(xì)介紹,socket可以建立連接,傳遞數(shù)據(jù),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下。
    2017-03-03
  • Pandas檢查dataFrame中的NaN實(shí)現(xiàn)

    Pandas檢查dataFrame中的NaN實(shí)現(xiàn)

    本文主要介紹了Pandas檢查dataFrame中的NaN實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-01-01
  • Python創(chuàng)建7種不同的文件格式的方法總結(jié)

    Python創(chuàng)建7種不同的文件格式的方法總結(jié)

    今天的這篇文章呢,小編來介紹一下如何通過Python來創(chuàng)建各種形式的文件,這里包括了:文本文件、CSV文件、Excel文件、壓縮文件、XML文件、JSON文件和PDF文件,需要的可以參考一下
    2023-01-01

最新評論