————————————————————————————————————————————————————————————

一.程序和請求上下文

  Flask從客戶端收到請求時,要讓視圖函數能訪問一些對象,這樣才能處理請求。請求對象就是一個很好的例子,它封裝了客戶端發送的Http請求。

  要想讓視圖函數能夠訪問請求對象,一個顯而易見的方式是將其作爲參數傳入視圖函數,不過這會導致程序中的每個視圖函數都增加一個參數。除了訪問請求對象,如果視圖函數在處理請求時還要訪問其他對象,情況會變得糟糕。

  爲了避免大量可有可無的參數,Flask使用了上下文臨時把某些對象變爲全局可訪問。有類上下文,就可以寫出下面的函數視圖。

 from flask import Flask
from flask import request
app = Flask(__name__) @app.route('/')
def index():
user_agent = request.headers.get('User-Agent')
return '<h3>Your browser is %s</h3>' %user_agent if __name__=='__main__':
app.run()

  在Flask中有兩種上下文:程序上下文和請求上下文。

  程序上下文:       current_app     當前激活程序的程序實例

          g                      處理請求時臨時用作臨時存儲對象。每次請求都會重置這個變量

  請求上下文:  request    請求對象,封裝了客戶端發出的HTTP請求中的內容

          session    用戶會話,用與存儲請求之間需要“記住”的值的詞典

  Flask在發送請求之前激活程序和請求上下文,請求處理完成後再將其刪除。

  下面這個Python shell會話演示了程序上下文的使用方法:

  >>>from hello import app

  >>>from flask import current_app

  >>> current_app.name

  Traceback (most recent call last):

  ...

  RuntimeError:working outside of application context

  >>> app_ctx = app.app_context()

  >>> app_ctx.push()

  >>> current_app.name

  'hello'

  >>> app_ctx.pop()

  在沒有激活程序上下文之前就調用current_app.name會導致錯誤,但推送完上下文之後就可以調用了。

  注意:在程序實例中調用app.app_context()可獲得一個程序上下文。

————————————————————————————————————————————————————————————

二.請求調度

  程序收到客戶端發來的請求時,要找到處理該請求的視圖函數。爲了完成這個任務,Flask會在程序的URL映射中查找請求的URL。URL映射是URL和視圖函數之間的對應關系。Flask使用app.route修飾器或者非修飾器形式的app.add_url_rule()生成映射。

  URL映射中的HEAD,Options,GET是請求方法,由路由進行處理。Flask爲每一個URL都指定了請求方法,這樣不同的請求方法發送到相同的URL上時,會使用不同的視圖函數進行處理。HEAD和OPTIONS方法由Flask自動處理,其他都書GET方法。

三.請求鉤子

  有時在處理請求之前或之後執行代碼會很有用。例如,在請求開始時,我們可能需要創建數據庫連接或者認證發起請求的用戶。爲了避免在每個視圖中都使用重復的代碼,Flask提供了注冊通用函數的功能,注冊函數可在請求被分發到視圖函數之前或之後調用。

  請求鉤子使用修飾器實現,Flask支持以下4種鉤子。

  *before_first_request : 注冊一個函數,在處理第一個請求之前運行。

  *before_request         : 注冊一個函數,在每次請求之前運行。

  *after_request            : 注冊一個函數,如果沒有處理異常的拋出,在每次請求之後運行。

  *teardown_request  : 注冊一個函數,即使有未處理的異常拋出,也在每次請求之後運行。

  在請求鉤子函數和視圖函數之間共享數據一般使用上下文全局變量g。例如,before_request處理程序可以從數據庫中加載已登錄用戶,並將其保存到g.user中。隨後調用視圖函數時,視圖函數再使用g.user獲取用戶。

四.响应

  Flask調用視圖函數後,會將其返回值作爲响应的內容。大多數情況下,响应就是一個簡單的字符串,作爲HTML頁面送回客戶端。

但HTTP協議需要的不僅是作爲請求响应的字符串。HTTP响应中一個很重要的部分是狀態碼,Flask默認設置是200,這個代碼表明請求已經被成功處理。

  Flask還可以返回Response對象。make_response()函數可以接受1個,2個或3個參數。並返回一個Response對象。有時我們需要在視圖函數中進行這種轉換,然後在響應對象上調用各種方法,進一步設置響應。

 from flask import Flask
from flask import make_response app = Flask(__name__) @app.route('/')
def index():
response = make_response('<h1>This is document</h1>')
response.set_cookie('answer','yooooooo~')
return response if __name__=='__main__':
app.run()

  有一種名爲重定向的特殊响应類型。這種响应沒有頁面文檔,只告訴瀏覽器一個新地址用以加載新頁面。重定向經常在Web表單中使用。狀態碼一般是302,指向的地址由Location首部提供。重定向响应可以使用3個形式的返回值生成,也可以在Response對象中設定。不過,由於使用頻繁,Flask提供了redirect()輔助函數,用於生成這種響應:

 from flask import Flask
from flask import redirect app = Flask(__name__) @app.route('/')
def index():
return redirect('http://www.baidu.com') if __name__=='__main__':
app.run()

    

  重定向至http://www.baidu.com

  還有一種特殊响应由abort函數生成,用於處理錯誤。如果URL中動態參數ID對應用戶不存在,就返回狀態碼404: 

 from flask import Flask
from flask import abort app = Flask(__name__) @app.route('/')
def index():
return 'hello' @app.route('/user/<id>')
def user(id):
user = load_user(id)
if not user:
abort(404)
return 'hello,%s' %user.name if __name__=='__main__':
app.run()

五.Flask扩展

 from flask import Flask

 app = Flask(__name__)

 @app.route('/')
def index():
return '<h7>hello</h7>' if __name__=='__main__':
app.run(host='192.168.1.8')

  之後便可在統一網絡下使用http://192.168.1.8:5000 來進行訪問。

  

Flask從入門到入土(二)——請求响应與Flask扩展的更多相关文章

  1. Flask從入門到入土(三)——模板

    模板是一個包含響應文本的文件,其中包含佔位變量表示的動態部分,其具體值只是請求上下文中才能知道.使用真實值替換變量,再返回最終得到的響應字符串,這一過程稱爲渲染.爲了渲染模板,Flask使用了一個名爲 ...

  2. Flask從入門到入土(三)——Web表單

    Flask-WTF擴展可以把處理Web表單的過程變成一種愉悅的體驗.這個擴展對獨立的WTForms包進行了包裝,方便集成到Flask程序中. Flask-WTF及其依賴可使用pip安裝: pip in ...

  3. Flask從入門到入土(一)——程序的基本結構

    一.初始化 所有Flask程序都必須創建一個程序實例.Web服務器使用一種名爲Web服務器網關接口的協議,把接收自客戶端的所有請求都轉交給這個對象處理.程序實例書Flask類的對象,創建代碼: fro ...

  4. Flask從入門到入土(四)——登錄實現

    表單介紹 1.表單標籤 聲明表單的範圍,位於表單標籤中的元素將被提交 語法: <form>  </form> 屬性: Method(提交方式get,post) , Enctyp ...

  5. Flask從入門到入土(五)——Flask与数据库

    import sqlite3 def init_db(): sql = 'create table user(id INT,name TEXT)' conn = sqlite3.connect(&qu ...

  6. GOOGLE搜索從入門到精通V4.0

    1,前言2,摘要3,如何使用本文4,Google簡介5,搜索入門6,初階搜索 6.1,搜索結果要求包含兩個及兩個以上關鍵字 6.2,搜索結果要求不包含某些特定資訊 6.3,搜索結果至少包含多個關鍵字中 ...

  7. 依賴注入入門——Unity(二)

    參考博客文章http://www.cnblogs.com/kebixisimba/category/130432.html http://www.cnblogs.com/qqlin/tag/Unity ...

  8. Delphi APP 開發入門(二)Android/iOS設定,Hello World

    Delphi APP 開發入門(二)Android/iOS設定,Hello World 分享: Share on facebookShare on twitterShare on google_plu ...

  9. Delphi APP 開發入門(一)重生的 Delphi

    Delphi APP 開發入門(一)重生的 Delphi 分享: Share on facebookShare on twitterShare on google_plusone_share   閲讀 ...

随机推荐

  1. jQuery给input绑定回车事件

    dataInput为Span元素ID <script type="text/javascript" src="Scripts/jquery-1.6.2.js&quo ...

  2. HTML学习(一)

    文本输出/超链接 <!--/* * @<h1></h1>到<h6></h6>六个h标签,分别表示不同大小的字体.h1最大,h6最小 * @< ...

  3. Windows核心编程&作业

    1. 作业内核对象 允许将进程组合在一起并创建一个"沙箱"来限制进程能够做什么.我们可以将作业内核对象想象成一个进程容器(即使只有一个进程也具有相当的重要性) 限制包括可以分配的最 ...

  4. 无废话XML--XML细节

    今天开始研究xml,其实在实际的开发中,我们参与到真正的XML开发并不是很多,最多写一个配置,但是我还是觉得很有必要把XML的知识整理一遍.作为基本的2种的数据交互载体(还有一个是json),基本的X ...

  5. 企业级分布式存储应用与实战FastDFS实现

    FASTDFS是什么 FastDFS是由国人余庆所开发,其项目地址:https://github.com/happyfish100 FastDFS是一个轻量级的开源分布式文件系统,主要解决了大容量的文 ...

  6. 本篇将记录python开发过程中常见问题

    1.Django 外键on_delete的使用 之前用的Django1.8 换成2.0后,出现以下问题: TypeError: init() missing 1 required positional ...

  7. docker入门(二)容器与镜像的理解

    10张图带你深入理解Docker容器和镜像 申明:此篇文章是转载的(原文地址http://dockone.io/article/783),今天意外发现已经有人转载了(复制了),希望大家关注原创 原本打 ...

  8. 关系类型字段 -- Django从入门到精通系列教程

    该系列教程系个人原创,并完整发布在个人官网刘江的博客和教程 所有转载本文者,需在顶部显著位置注明原作者及www.liujiangblog.com官网地址. Python及Django学习QQ群:453 ...

  9. 封装的应用【example_Array工具】

    定义一个数组工具[ArrayTool]封装其方法,ArrayDemo调用数组工具ArrayTool package new_Object; //封装多个个功能 class ArrayTool{ //1 ...

  10. java中线程的状态详解

    一.线程的五种状态   线程的生命周期可以大致分为5种,但这种说法是比较旧的一种说法,有点过时了,或者更确切的来说,这是操作系统的说法,而不是java的说法.但对下面所说的六种状态的理解有所帮助,所以 ...