1. 程式人生 > 程式設計 >python manage.py runserver流程解析

python manage.py runserver流程解析

這篇文章主要介紹了python manage.py runserver流程解析,文中通過示例程式碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下

版本

  • python27
  • django 1.0

搭建可執行的環境

建立python27 虛擬環境

github 下載 django-1.0.tar.gz(1.0 版本的django)

解壓

可以看到,有個 demo 在 examples 目錄

把 django 目錄拷貝到 examples 下面,這樣 example 可以正確匯入 django1.0

啟動專案

python manage.py runserver

專案啟動成功,可以修改程式碼來跟蹤執行流程

流程

以下程式碼存在刪減,主要展示程式碼流程

從 manage.py 開始,執行了 execute_manager 方法,傳入 settings 模組

execute_manager(settings)

django.core.management.execute_manager 方法

def execute_manager(settings_mod,argv=None):
  # setup_environ 函式,只是設定了環境變數,執行配置模組
  # os.environ['DJANGO_SETTINGS_MODULE'] = examples.settting
  setup_environ(settings_mod)

  # admin manage 工具類
  utility = ManagementUtility(argv)
  utility.execute()

ManagementUtility 類

class ManagementUtility(object):
  def __init__(self,argv=None):
    # 初始化,例如
    self.argv = ['.../examples/manage.py','runserver']
    self.prog_name = 'manage.py'
  def execute(self):
    # 刪除了部分程式碼,最終執行程式碼大致如下
  
    # 這是一個命令列工具類,表名能接受什麼樣的引數,這裡主要檢查兩個引數
    # --settings 指定配置檔案
    # --pythonpath 執行 python 環境變數
    parser = LaxOptionParser(usage="%prog subcommand [options] [args]",version=get_version(),option_list=BaseCommand.option_list)
  
    # 使用命令列工具類解析命令列引數,也就是獲取 --settings 和 --pythonpath 的引數值
    options,args = parser.parse_args(self.argv)
    # 如果 --settings 引數存在,會覆蓋之前設定的 os.environ['DJANGO_SETTINGS_MODULE']
    # 如果 --pythonpath 引數存在,會把指定路徑新增到 sys.path 的第一位,優先從此處載入模組
    handle_default_options(options)
  
    # fetch_command
    # fetch_command 分析在下邊
    # fetch_command 返回 django.core.management.commands.runserver.Command
    # run_from_argv
    # run_from_argv 分析在下邊
    self.fetch_command(subcommand).run_from_argv(self.argv)
  
  def fetch_command(self,subcommand):
    # get_commands
    # get_commands 返回 django.core.management.commands 目錄下的所有模組,每個模組處理對應的引數
    # 每個模組的值都是 django.core,app_name = 'django.core'
    app_name = get_commands()[subcommand]
  
    # load_command_class 方法
    # 返回了 django.core.management.commands.runserver.Command
    klass = load_command_class(app_name,subcommand)
  
    return klass

run_from_argv 方法

# django.core.management.commands.runserver.Command 
# 繼承 django.core.management.base import BaseCommand
# run_from_argv 也是繼承的
def run_from_argv(self,argv):
  # 呼叫 execute
  self.execute(*args,**options.__dict__)

def execute(self,*args,**options):
  # 呼叫 handle
  # 注意 handle 被重寫了
  # 呼叫的是 django.core.management.commands.runserver.Command.handle
  output = self.handle(*args,**options)

handle

def handle(self,addrport='',**options):
  def inner_run():
    # WSGI 處理程式
    # WSGIHandler 可呼叫,是 WSGI 處理程式
    # AdminMediaHandler 是對 WSGIHandler 的封裝
    # AdminMediaHandler 特殊處理媒體檔案請求
    # AdminMediaHandler 非媒體檔案的 HTTP 請求,直接返回 WSGIHandler
    handler = AdminMediaHandler(WSGIHandler(),path)
    
    # 
    run(addr,int(port),handler)
    # run 在 django.core.servers.basehttp.run
    # run 定義如下
    # run 啟動了 HTTP 服務,這個伺服器只能用於開發除錯
    def run(addr,port,wsgi_handler):
      # 繫結地址埠
      server_address = (addr,port)
      # 服務例項
      httpd = WSGIServer(server_address,WSGIRequestHandler)
      # 傳入 WSGI 處理程式
      httpd.set_app(wsgi_handler)
      # 監聽請求
      httpd.serve_forever()
  
  inner_run()

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支援我們。