Flask正則路由,異常捕獲和請求鈎子


正則匹配路由

在 web 開發中,可能會出現限制用戶訪問規則的場景,那么這個時候就需要用到正則匹配,根據自己的規則去限定請求參數再進行訪問

具體實現步驟為:

  • 導入轉換器基類:在 Flask 中,所有的路由的匹配規則都是使用轉換器對象進行記錄
  • 自定義轉換器:自定義類繼承於轉換器基類
  • 添加轉換器到默認的轉換器字典中
  • 使用自定義轉換器實現自定義匹配規則

代碼實現

  • 導入轉換器基類
from werkzeug.routing import BaseConverter 
  • 自定義轉換器
# 自定義正則轉換器 class RegexConverter(BaseConverter): def __init__(self, url_map, *args): super(RegexConverter, self).__init__(url_map) # 將接受的第1個參數當作匹配規則進行保存 self.regex = args[0] 
  • 添加轉換器到默認的轉換器字典中,並指定轉換器使用時名字為: re
app = Flask(__name__)

# 將自定義轉換器添加到轉換器字典中,並指定轉換器使用時名字為: re app.url_map.converters['re'] = RegexConverter 
  • 使用轉換器去實現自定義匹配規則
    • 當前此處定義的規則是:3位數字
@app.route('/user/<re("[0-9]{3}"):user_id>') def user_info(user_id): return "user_id 為 %s" % user_id 

運行測試:http://127.0.0.1:5000/user/123 ,如果訪問的url不符合規則,會提示找不到頁面

自定義轉換器其他兩個函數實現

繼承於自定義轉換器之后,還可以實現 to_python 和 to_url 這兩個函數去對匹配參數做進一步處理:

  • to_python:
    • 該函數參數中的 value 值代表匹配到的值,可輸出進行查看
    • 匹配完成之后,對匹配到的參數作最后一步處理再返回,比如:轉成 int 類型的值再返回:
class RegexConverter(BaseConverter): def __init__(self, url_map, *args): super(RegexConverter, self).__init__(url_map) # 將接受的第1個參數當作匹配規則進行保存 self.regex = args[0] def to_python(self, value): return int(value) 

運行測試,在視圖函數中可以查看參數的類型,由之前默認的 str 已變成 int 類型的值

  • to_url:
    • 在使用 url_for 去獲取視圖函數所對應的 url 的時候,會調用此方法對 url_for 后面傳入的視圖函數參數做進一步處理
    • 具體可參見 Flask 的 app.py 中寫的示例代碼:ListConverter

系統自帶轉換器

DEFAULT_CONVERTERS = {
    'default': UnicodeConverter, 'string': UnicodeConverter, 'any': AnyConverter, 'path': PathConverter, 'int': IntegerConverter, 'float': FloatConverter, 'uuid': UUIDConverter, }
系統自帶的轉換器具體使用方式在每種轉換器的注釋代碼中有寫,請留意每種轉換器初始化的參數。

異常捕獲

HTTP 異常主動拋出

  • abort 方法
    • 拋出一個給定狀態代碼的 HTTPException 或者 指定響應,例如想要用一個頁面未找到異常來終止請求,你可以調用 abort(404)。
  • 參數:
    • code – HTTP的錯誤狀態碼
# abort(404) abort(500) 

拋出狀態碼的話,只能拋出 HTTP 協議的錯誤狀態碼

 

捕獲錯誤

  • errorhandler 裝飾器
    • 注冊一個錯誤處理程序,當程序拋出指定錯誤狀態碼的時候,就會調用該裝飾器所裝飾的方法
  • 參數:
    • code_or_exception – HTTP的錯誤狀態碼或指定異常
  • 例如統一處理狀態碼為500的錯誤給用戶友好的提示:
@app.errorhandler(500) def internal_server_error(e): return '服務器搬家了' 
  • 捕獲指定異常
@app.errorhandler(ZeroDivisionError) def zero_division_error(e): return '除數不能為0'

可以全局捕獲404錯誤,返回渲染的404頁面.

請求勾子(相當於Django中的中間件)

在客戶端和服務器交互的過程中,有些准備工作或掃尾工作需要處理,比如:

  • 在請求開始時,建立數據庫連接;
  • 在請求開始時,根據需求進行權限校驗;
  • 在請求結束時,指定數據的交互格式;

為了讓每個視圖函數避免編寫重復功能的代碼,Flask提供了通用設施的功能,即請求鈎子。

請求鈎子是通過裝飾器的形式實現,Flask支持如下四種請求鈎子:

  • before_first_request
    • 在處理第一個請求前執行
  • before_request
    • 在每次請求前執行
    • 如果在某修飾的函數中返回了一個響應,視圖函數將不再被調用
  • after_request
    • 如果沒有拋出錯誤,在每次請求后執行
    • 接受一個參數:視圖函數作出的響應
    • 在此函數中可以對響應值在返回之前做最后一步修改處理
    • 需要將參數中的響應在此參數中進行返回
  • teardown_request:
    • 在每次請求后執行
    • 接受一個參數:錯誤信息,如果有相關錯誤拋出

代碼測試

from flask import Flask from flask import abort app = Flask(__name__) # 在第一次請求之前調用,可以在此方法內部做一些初始化操作 @app.before_first_request def before_first_request(): print("before_first_request") # 在每一次請求之前調用,這時候已經有請求了,可能在這個方法里面做請求的校驗 # 如果請求的校驗不成功,可以直接在此方法中進行響應,直接return之后那么就不會執行視圖函數 @app.before_request def before_request(): print("before_request") # if 請求不符合條件: # return "laowang" # 在執行完視圖函數之后會調用,並且會把視圖函數所生成的響應傳入,可以在此方法中對響應做最后一步統一的處理 @app.after_request def after_request(response): print("after_request") response.headers["Content-Type"] = "application/json" return response # 請每一次請求之后都會調用,會接受一個參數,參數是服務器出現的錯誤信息 @app.teardown_request def teardown_request(e): print("teardown_request") @app.route('/') def index(): return 'index' if __name__ == '__main__': app.run(debug=True) 
  • 在第1次請求時的打印:
before_first_request
before_request
after_request
teardown_request
  • 在第2次請求時的打印:
before_request
after_request
teardown_request





免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM