Flask是一個(gè)Python編寫(xiě)的Web 微框架,讓我們可以使用Python語(yǔ)言快速實(shí)現(xiàn)一個(gè)網(wǎng)站或Web服務(wù)。本文參考自Flask官方文檔,大部分代碼引用自官方文檔。
安裝Flask
首先我們來(lái)安裝Flask。最簡(jiǎn)單的辦法就是使用pip。
pip install flask
然后打開(kāi)一個(gè)Python文件,輸入下面的內(nèi)容并運(yùn)行該文件。然后訪問(wèn)localhost:5000,我們應(yīng)當(dāng)可以看到瀏覽器上輸出了Hello Flask!。
from flask import Flaskapp = Flask(__name__)@app.route('/')def hello_world(): return 'Hello Flask!'if __name__ == '__main__': app.run()
快速開(kāi)始
調(diào)試模式
我們修改代碼中的輸出,然后查看瀏覽器上是否有變化。如果你照做的話,可以看到什么變化都沒(méi)有。其實(shí)Flask內(nèi)置了調(diào)試模式,可以自動(dòng)重載代碼并顯示調(diào)試信息。這需要我們開(kāi)啟調(diào)試模式,方法很簡(jiǎn)單,設(shè)置FLASK_DEBUG環(huán)境變量,并將值設(shè)置為1。
然后再次運(yùn)行程序,會(huì)看到有這樣的輸出。這時(shí)候如果再次修改代碼,會(huì)發(fā)現(xiàn)這次Flask會(huì)自動(dòng)重啟。
* Restarting with stat* Debugger is active!* Debugger PIN: 157-063-180* Running on http://127.0.0.1:5000/ (Press CTRL+C to quit)
路由
在上面的例子里可以看到路由的使用。如果了解Spring Web MVC的話,應(yīng)該對(duì)路由很熟悉。路由通過(guò)使用Flask的app.route裝飾器來(lái)設(shè)置,這類(lèi)似Java的注解。
@app.route('/')def index(): return 'Index Page'@app.route('/hello')def hello(): return 'Hello, World'
路徑變量
如果希望獲取/article/1這樣的路徑參數(shù),就需要使用路徑變量。路徑變量的語(yǔ)法是/path/
轉(zhuǎn)換器
下面是Flask官方的例子。
@app.route('/user/
構(gòu)造URL
在Web程序中常常需要獲取某個(gè)頁(yè)面的URL,在Flask中需要使用url_for('方法名')來(lái)構(gòu)造對(duì)應(yīng)方法的URL。下面是Flask官方的例子。
>>> from flask import Flask, url_for>>> app = Flask(__name__)>>> @app.route('/')... def index(): pass...>>> @app.route('/login')... def login(): pass...>>> @app.route('/user/
HTTP方法
如果需要處理具體的HTTP方法,在Flask中也很容易,使用route裝飾器的methods參數(shù)設(shè)置即可。
from flask import request@app.route('/login', methods=['GET', 'POST'])def login(): if request.method == 'POST': do_the_login() else: show_the_login_form()
靜態(tài)文件
Web程序中常常需要處理靜態(tài)文件,在Flask中需要使用url_for函數(shù)并指定static端點(diǎn)名和文件名。在下面的例子中,實(shí)際的文件應(yīng)放在static/文件夾下。
url_for('static', filename='style.css')
模板生成
Flask默認(rèn)使用Jinja2作為模板,F(xiàn)lask會(huì)自動(dòng)配置Jinja 模板,所以我們不需要其他配置了。默認(rèn)情況下,模板文件需要放在templates文件夾下。
使用 Jinja 模板,只需要使用render_template函數(shù)并傳入模板文件名和參數(shù)名即可。
from flask import render_template@app.route('/hello/')@app.route('/hello/
相應(yīng)的模板文件如下。
Hello {{ name }}!
{% else %}Hello, World!
{% endif %}日志輸出
Flask 為我們預(yù)配置了一個(gè) Logger,我們可以直接在程序中使用。這個(gè)Logger是一個(gè)標(biāo)準(zhǔn)的Python Logger,所以我們可以向標(biāo)準(zhǔn)Logger那樣配置它,詳情可以參考官方文檔或者我的文章Python 日志輸出。
app.logger.debug('A value for debugging')app.logger.warning('A warning occurred (%d apples)', 42)app.logger.error('An error occurred')
處理請(qǐng)求
在 Flask 中獲取請(qǐng)求參數(shù)需要使用request等幾個(gè)全局對(duì)象,但是這幾個(gè)全局對(duì)象比較特殊,它們是Context Locals,其實(shí)就是 Web 上下文中局部變量的代理。雖然我們?cè)诔绦蛑惺褂玫氖侨肿兞浚菍?duì)于每個(gè)請(qǐng)求作用域,它們都是互不相同的變量。理解了這一點(diǎn),后面就非常簡(jiǎn)單了。
Request 對(duì)象
Request 對(duì)象是一個(gè)全局對(duì)象,利用它的屬性和方法,我們可以方便的獲取從頁(yè)面?zhèn)鬟f過(guò)來(lái)的參數(shù)。
method屬性會(huì)返回HTTP方法的類(lèi)似,例如post和get。form屬性是一個(gè)字典,如果數(shù)據(jù)是POST類(lèi)型的表單,就可以從form屬性中獲取。下面是 Flask 官方的例子,演示了 Request 對(duì)象的method和form屬性。
from flask import request@app.route('/login', methods=['POST', 'GET'])def login(): error = None if request.method == 'POST': if valid_login(request.form['username'], request.form['password']): return log_the_user_in(request.form['username']) else: error = 'Invalid username/password' # the code below is executed if the request method # was GET or the credentials were invalid return render_template('login.html', error=error)
如果數(shù)據(jù)是由GET方法傳送過(guò)來(lái)的,可以使用args屬性獲取,這個(gè)屬性也是一個(gè)字典。
searchword = request.args.get('key', '')
文件上傳
利用Flask也可以方便的獲取表單中上傳的文件,只需要利用 request 的files屬性即可,這也是一個(gè)字典,包含了被上傳的文件。如果想獲取上傳的文件名,可以使用filename屬性,不過(guò)需要注意這個(gè)屬性可以被客戶端更改,所以并不可靠。更好的辦法是利用werkzeug提供的secure_filename方法來(lái)獲取安全的文件名。
from flask import requestfrom werkzeug.utils import secure_filename@app.route('/upload', methods=['GET', 'POST'])def upload_file(): if request.method == 'POST': f = request.files['the_file'] f.save('/var/www/uploads/' + secure_filename(f.filename))
Cookies
Flask也可以方便的處理Cookie。使用方法很簡(jiǎn)單,直接看官方的例子就行了。下面的例子是如何獲取cookie。
from flask import request@app.route('/')def index(): username = request.cookies.get('username') # 使用 cookies.get(key) 代替 cookies[key] 避免 # 得到 KeyError 如果cookie不存在
如果需要發(fā)送cookie給客戶端,參考下面的例子。
from flask import make_response@app.route('/')def index(): resp = make_response(render_template(...)) resp.set_cookie('username', 'the username') return resp
重定向和錯(cuò)誤
redirect和abort函數(shù)用于重定向和返回錯(cuò)誤頁(yè)面。
from flask import abort, redirect, url_for@app.route('/')def index(): return redirect(url_for('login'))@app.route('/login')def login(): abort(401) this_is_never_executed()
默認(rèn)的錯(cuò)誤頁(yè)面是一個(gè)空頁(yè)面,如果需要自定義錯(cuò)誤頁(yè)面,可以使用errorhandler裝飾器。
from flask import render_template@app.errorhandler(404)def page_not_found(error): return render_template('page_not_found.html'), 404
響應(yīng)處理
默認(rèn)情況下,F(xiàn)lask會(huì)根據(jù)函數(shù)的返回值自動(dòng)決定如何處理響應(yīng):如果返回值是響應(yīng)對(duì)象,則直接傳遞給客戶端;如果返回值是字符串,那么就會(huì)將字符串轉(zhuǎn)換為合適的響應(yīng)對(duì)象。我們也可以自己決定如何設(shè)置響應(yīng)對(duì)象,方法也很簡(jiǎn)單,使用make_response函數(shù)即可。
@app.errorhandler(404)def not_found(error): resp = make_response(render_template('error.html'), 404) resp.headers['X-Something'] = 'A value' return resp
Sessions
我們可以使用全局對(duì)象session來(lái)管理用戶會(huì)話。Sesison 是建立在 Cookie 技術(shù)上的,不過(guò)在 Flask 中,我們還可以為 Session 指定密鑰,這樣存儲(chǔ)在 Cookie 中的信息就會(huì)被加密,從而更加安全。直接看 Flask 官方的例子吧。
from flask import Flask, session, redirect, url_for, escape, requestapp = Flask(__name__)@app.route('/')def index(): if 'username' in session: return 'Logged in as %s' % escape(session['username']) return 'You are not logged in'@app.route('/login', methods=['GET', 'POST'])def login(): if request.method == 'POST': session['username'] = request.form['username'] return redirect(url_for('index')) return '''
'''@app.route('/logout')def logout(): # remove the username from the session if it's there session.pop('username', None) return redirect(url_for('index'))# set the secret key. keep this really secret:app.secret_key = 'A0Zr98j/3yX R~XHH!jmN]LWX/,?RT'模板簡(jiǎn)介
這里簡(jiǎn)單的介紹一下Jinja 模板的使用方法,詳細(xì)資料直接看原文檔吧。
模板標(biāo)簽
其實(shí)Jinja 模板和其他語(yǔ)言和框架的模板類(lèi)似,反正都是通過(guò)某種語(yǔ)法將HTML文件中的特定元素替換為實(shí)際的值。如果使用過(guò)JSP、Thymeleaf 等模板,應(yīng)該可以非常容易的學(xué)會(huì)使用 Jinja模板。
其實(shí)從上面的例子中我們應(yīng)該可以看到Jinja 模板的基本語(yǔ)法了。代碼塊需要包含在{% %}塊中,例如下面的代碼。
{% extends 'layout.html' %}{% block title %}主頁(yè){% endblock %}{% block body %}
主頁(yè)
雙大括號(hào)中的內(nèi)容不會(huì)被轉(zhuǎn)義,所有內(nèi)容都會(huì)原樣輸出,它常常和其他輔助函數(shù)一起使用。下面是一個(gè)例子。
繼承
模板可以繼承其他模板,我們可以將布局設(shè)置為父模板,讓其他模板繼承,這樣可以非常方便的控制整個(gè)程序的外觀。
例如這里有一個(gè)layout.html模板,它是整個(gè)程序的布局文件。
其他模板可以這么寫(xiě)。對(duì)比一下面向?qū)ο?a href="http://ttokpm.com/v/tag/1315/" target="_blank">編程的繼承概念,我們可以很容易的理解。
{% extends 'layout.html' %}{% block title %}主頁(yè){% endblock %}{% block body %}
主頁(yè)
本項(xiàng)目演示了Flask的簡(jiǎn)單使用方法,點(diǎn)擊導(dǎo)航欄上的菜單條查看具體功能。
控制流
條件判斷可以這么寫(xiě),類(lèi)似于JSP標(biāo)簽中的Java 代碼,{% %}中也可以寫(xiě)Python代碼。下面是Flask官方文檔的例子。
循環(huán)的話可以這么寫(xiě),和在Python中遍歷差不多。
{% for key,value in data.items() %}
需要注意不是所有的Python代碼都可以寫(xiě)在模板里,如果希望從模板中引用其他文件的函數(shù),需要顯式將函數(shù)注冊(cè)到模板中??梢詤⒖歼@個(gè)爆棧提問(wèn)。
寫(xiě)在最后
這篇文章主要參考了Flask的官方文檔,但是只介紹了 Flask的最基本的一部分。了解了這部分,我們可以用Python 搭一個(gè)小服務(wù)器做點(diǎn)事情。如果希望詳細(xì)了解 Flask的使用用法,請(qǐng)關(guān)注更詳細(xì)的資料。本文就是起一個(gè)拋磚引玉的效果。
順便說(shuō),通過(guò)Flask 我也了解了Python 語(yǔ)言的執(zhí)行速度。我們都知道編譯器編譯出來(lái)的代碼執(zhí)行起來(lái)要比解釋器解釋代碼要快大約幾十倍到幾千倍不等。以前學(xué)Java的時(shí)候,感覺(jué)Java 慢,主要原因就是等待編譯時(shí)間比較長(zhǎng)。相對(duì)來(lái)說(shuō)用Python寫(xiě)腳本就很塊了,因?yàn)闆](méi)有編譯過(guò)程。
但是從Flask的運(yùn)行速度來(lái)看,我切身感受到了Python 執(zhí)行確實(shí)不快。舉個(gè)例子,在Spring中寫(xiě)一個(gè)控制器,接受HTTP參數(shù),并顯示到頁(yè)面上,如果程序編譯完之后,這個(gè)顯示過(guò)程基本是瞬時(shí)的。但是同樣的需求在Flask中,我居然可以感覺(jué)到明顯的延遲(大概幾百毫秒的等待時(shí)間)。所以,如果你想寫(xiě)一個(gè)比較快的Web程序,還是用Java或者JVM語(yǔ)言吧,雖然看著土,性能確實(shí)杠杠的 。
-
JAVA
+關(guān)注
關(guān)注
19文章
2952瀏覽量
104489 -
python
+關(guān)注
關(guān)注
55文章
4768瀏覽量
84376 -
flask
+關(guān)注
關(guān)注
0文章
16瀏覽量
3603
原文標(biāo)題:3000 字 Flask 快速學(xué)習(xí)指南:從入門(mén)到開(kāi)發(fā)
文章出處:【微信號(hào):magedu-Linux,微信公眾號(hào):馬哥Linux運(yùn)維】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。
發(fā)布評(píng)論請(qǐng)先 登錄
相關(guān)推薦
評(píng)論