使用Python的Flask框架來(lái)搭建第一個(gè)Web應(yīng)用程序
1、初始化
在這章,你將學(xué)到Flask應(yīng)用程序的不同部分。同時(shí),你將編寫和運(yùn)行你的第一個(gè)Flask web應(yīng)用程序。
所有的Flask應(yīng)用程序都必須創(chuàng)建一個(gè) 應(yīng)用程序?qū)嵗?。使用web服務(wù)器網(wǎng)關(guān)接口協(xié)議將所有從客戶端接收的請(qǐng)求傳遞給這個(gè)對(duì)象處理。這個(gè)應(yīng)用程序?qū)嵗褪荈lask類的一個(gè)對(duì)象,通常使用下面的方式創(chuàng)建:
from flask import Flask app = Flask(__name__)
Flask類構(gòu)造函數(shù)唯一需要的參數(shù)就是應(yīng)用程序的主模塊或包。對(duì)于大多數(shù)應(yīng)用程序,Python的__name__變量就是那個(gè)正確的、你需要傳遞的值。
注:對(duì)于Flask開(kāi)發(fā)者來(lái)說(shuō),傳給Flask應(yīng)用程序構(gòu)造函數(shù)的name參數(shù)是比較容易弄混淆的。Flask使用這個(gè)參數(shù)來(lái)確定應(yīng)用程序的根目錄,這樣以后可以相對(duì)這個(gè)路徑來(lái)找到資源文件。
稍后你可以看到更復(fù)雜的應(yīng)用程序?qū)嵗跏蓟?,但是?duì)于簡(jiǎn)單應(yīng)用程序這些已經(jīng)足夠了。
2、路由和視圖函數(shù)
客戶端例如web瀏覽器發(fā)送 請(qǐng)求 給web服務(wù),進(jìn)而將它們發(fā)送給Flask應(yīng)用程序?qū)嵗?。?yīng)用程序?qū)嵗枰缹?duì)于各個(gè)URL請(qǐng)求需要運(yùn)行哪些代碼,所以它給Python函數(shù)建立了一個(gè)URLs映射。這些在URL和函數(shù)之間建立聯(lián)系的操作被稱之為 路由 。
在Flask應(yīng)程序中定義路由的最便捷的方式是通過(guò)顯示定義在應(yīng)用程序?qū)嵗系腶pp.route裝飾器,注冊(cè)被裝飾的函數(shù)來(lái)作為一個(gè)路由。下面的例子會(huì)演示怎樣使用裝飾器來(lái)申明一個(gè)路由:
@app.route('/') def index(): return '<h1>Hello World!</h1>'
注:裝飾器是Python語(yǔ)言的標(biāo)準(zhǔn)特性;它們可以以不同方式改變函數(shù)的行為。一個(gè)常見(jiàn)的模式是使用裝飾器來(lái)注冊(cè)函數(shù)作為一個(gè)事件處理程序。
在上一個(gè)示例給應(yīng)用程序的根URL注冊(cè)index()函數(shù)作為事件的處理程序。如果這個(gè)應(yīng)用程序被部署在服務(wù)器上并綁定了 www.example.com 域名,然后在你的瀏覽器地址欄中輸入 http://www.example.com 將觸發(fā)index()來(lái)運(yùn)行服務(wù)??蛻舳私邮盏降倪@個(gè)函數(shù)的返回值被稱為 響應(yīng) 。如果客戶端是web瀏覽器,響應(yīng)則是顯示給用戶的文檔。
類似于index()的函數(shù)被稱作 視圖函數(shù) 。通過(guò)視圖返回的響應(yīng)可以是簡(jiǎn)單的HTML內(nèi)容的字符串,但它也可以市更復(fù)雜的形式,正如您將看到的。
注:響應(yīng)字符串嵌入在Python代碼中導(dǎo)致代碼難以掌控,在此只是介紹響應(yīng)的概念。你將在第三章學(xué)習(xí)正確的方法來(lái)生成響應(yīng)。
如果你注意到你每天使用的一些網(wǎng)站URLs如何形成的,你將會(huì)發(fā)現(xiàn)很多都有變量。例如,你的Facebook個(gè)人信息頁(yè)的URL是 http://www.facebook.com/<username> ,所以你的用戶名是它的一部分。Flask在路由裝飾器中使用特殊的語(yǔ)法支持這些類型的URLs。下面的示例定義了一個(gè)擁有動(dòng)態(tài)名稱組件的路由:
@app.route('/user/<name>') def user(name): return '<h1>Hello, %s!</h1>' % name
用尖括號(hào)括起來(lái)的部分是動(dòng)態(tài)的部分,所以任何URLs匹配到靜態(tài)部分都將映射到這個(gè)路由。當(dāng)視圖函數(shù)被調(diào)用,F(xiàn)lask發(fā)送動(dòng)態(tài)組件作為一個(gè)參數(shù)。在前面的示例的視圖函數(shù)中,這個(gè)參數(shù)是用于生成一個(gè)個(gè)性的問(wèn)候作為響應(yīng)。
在路由中動(dòng)態(tài)組件默認(rèn)為字符串,但是可以定義為其他類型。例如,路由/user/<int:id>只匹配有一個(gè)整數(shù)在id動(dòng)態(tài)段的URLs。Flask路由支持int、float和path。path同樣是字符串類型,但并不認(rèn)為斜杠是分隔符,而認(rèn)為它們是動(dòng)態(tài)組件的一部分。
3、服務(wù)啟動(dòng)
應(yīng)用程序?qū)嵗幸粋€(gè)run方法用于啟動(dòng)Flask集成的web服務(wù):
if __name__ == '__main__': app.run(debug=True)
__name__ == '__main__'在此處使用是用于確保web服務(wù)已經(jīng)啟動(dòng)當(dāng)腳本被立即執(zhí)行。當(dāng)腳本被另一個(gè)腳本導(dǎo)入,它被看做父腳本將啟動(dòng)不同的服務(wù),所以app.run()調(diào)用會(huì)被跳過(guò)。
一旦服務(wù)啟動(dòng),它將進(jìn)入循環(huán)等待請(qǐng)求并為之服務(wù)。這個(gè)循環(huán)持續(xù)到應(yīng)用程序停止,例如通過(guò)按下Ctrl-C。
有幾個(gè)選項(xiàng)參數(shù)可以給app.run()配置web服務(wù)的操作模式。在開(kāi)發(fā)期間,可以很方便的開(kāi)啟debug模式,將激活 debugger 和 reloader 。這樣做是通過(guò)傳遞debug為True來(lái)實(shí)現(xiàn)的。
注:Flask提供的web服務(wù)并不用于生產(chǎn)環(huán)境。你將在十七章學(xué)習(xí)生產(chǎn)環(huán)境的web服務(wù)。
4、一個(gè)完整的應(yīng)用程序
在上一節(jié),你學(xué)習(xí)了Flask web應(yīng)用程序的不同部分,現(xiàn)在是時(shí)候?qū)懸粋€(gè)了。整個(gè) hello.py 應(yīng)用程序腳本只不過(guò)將前面描述的三個(gè)部分結(jié)合在一個(gè)文件中。應(yīng)用程序示例2-1所示。
示例 hello.py:一個(gè)完整的Flask應(yīng)用程序
from flask import Flask app = Flask(__name__) @app.route('/') def index(): return ' <h1>Hello World!</h1> ' if __name__ == '__main__': app.run(debug=True)
建議:如果你有克隆在GitHub上的應(yīng)用程序,你現(xiàn)在可以運(yùn)行g(shù)it checkout 2a來(lái)切換到這個(gè)版本的應(yīng)用程序。
運(yùn)行應(yīng)用程序之前,請(qǐng)確保你在之前創(chuàng)建的虛擬環(huán)境已經(jīng)是激活狀態(tài)且已安裝Flask。現(xiàn)在打開(kāi)你的web瀏覽器并在地址欄輸入 http://127.0.0.1:5000/ 。下圖顯示連接到應(yīng)用程序后的web瀏覽器。
然后輸入以下命令啟動(dòng)應(yīng)用程序:
(venv) $ python hello.py * Running on http://127.0.0.1:5000/ * Restarting with reloader
如果你輸入任何其他URL,應(yīng)用程序?qū)⒉恢廊绾尾僮魉⑶覍⒎祷劐e(cuò)誤代碼404給瀏覽器——當(dāng)你訪問(wèn)一個(gè)不存在的網(wǎng)頁(yè)也會(huì)得到該錯(cuò)誤。
下面所示應(yīng)用程序的增強(qiáng)版添加了第二個(gè)動(dòng)態(tài)路由。當(dāng)你訪問(wèn)這個(gè)URI,你應(yīng)該可以看到一個(gè)個(gè)性的問(wèn)候。
示例hello.py:帶有動(dòng)態(tài)路由的Flask應(yīng)用程序
from flask import Flask app = Flask(__name__) @app.route('/') def index(): return ' <h1>Hello World!</h1> ' @app.route('/user/<name>') def user(name): return ' <h1>Hello, %s!</h1> ' % name if __name__ == '__main__': app.run(debug=True)
建議:如果你有克隆在GitHub上的應(yīng)用程序,你現(xiàn)在可以運(yùn)行g(shù)it checkout 2b來(lái)切換到這個(gè)版本的應(yīng)用程序。
測(cè)試動(dòng)態(tài)路由,確保服務(wù)正在運(yùn)行隨后訪問(wèn) http://localhost:5000/user/Dave 。生成的應(yīng)用程序會(huì)使用動(dòng)態(tài)參數(shù)名響應(yīng)一個(gè)定制的問(wèn)候。嘗試不同的名稱,看看視圖函數(shù)總是生成響應(yīng)基于給定的名稱。
5、請(qǐng)求-響應(yīng)循環(huán)
現(xiàn)在你已經(jīng)玩過(guò)一個(gè)基本的Flask應(yīng)用程序,你也許想要知道更多關(guān)于Flask如何施展魔力。下面章節(jié)描述了一些框架設(shè)計(jì)方面的特點(diǎn)。
5.1、應(yīng)用程序Context和請(qǐng)求Context
當(dāng)Flask從客戶端收到一個(gè)請(qǐng)求,它需要提供幾個(gè)可用對(duì)象給視圖函數(shù)處理。request對(duì)象是個(gè)不錯(cuò)的例子,它封裝了客戶端發(fā)送的HTTP請(qǐng)求。
Flask視圖函數(shù)訪問(wèn)request對(duì)象的最好方式,就是作為一個(gè)參數(shù)發(fā)送它,但這需要每個(gè)單一視圖函數(shù)在應(yīng)用程序中有一個(gè)額外的參數(shù)??紤]一下,如果request對(duì)象不是唯一一個(gè)視圖函數(shù)需要訪問(wèn)完成請(qǐng)求的對(duì)象,事情將會(huì)變得更加復(fù)雜。
為了避免弄亂視圖函數(shù)那些可能需要或不需要的參數(shù),F(xiàn)lask使用context來(lái)臨時(shí)確定可訪問(wèn)的全局對(duì)象。也多虧了context,視圖函數(shù)可以寫成下面這樣:
from flask import request @app.route('/') def index(): user_agent = request.headers.get('User-Agent') return ' <p>Your browser is %s</p> ' % user_agent
注意,在這個(gè)視圖函數(shù)中,request是如何被作為一個(gè)全局變量來(lái)使用的。現(xiàn)實(shí)中,request是不能作為全局變量的,如果是多線程服務(wù)器,同一時(shí)間線程作用于不同客戶端的不同請(qǐng)求,所以每一個(gè)線程需要看到request中的不同對(duì)象。contexts使得Flask確定可訪問(wèn)的全局變量而不干擾其他線程。
注:線程是可以獨(dú)立管理的最小指令序列。一個(gè)進(jìn)程中有多個(gè)活動(dòng)的線程是非常常見(jiàn)的,有時(shí)分享內(nèi)存或文件句柄資源。多線程web服務(wù)器會(huì)啟動(dòng)一個(gè)線程池并從池中選擇一個(gè)線程來(lái)處理每個(gè)傳入的請(qǐng)求。
Flask有兩類context:應(yīng)用級(jí)context 和 請(qǐng)求級(jí)context。表2-1展示了這些context提供的變量。
Flask激活(或壓棧)應(yīng)用級(jí)context和請(qǐng)求級(jí)context在調(diào)度請(qǐng)求之前,然后刪除他們當(dāng)請(qǐng)求被處理后。當(dāng)應(yīng)用程序context被壓入棧,線程中current_app和g變量變得可用;同樣的,當(dāng)請(qǐng)求級(jí)context被壓入棧,request和session變量也同樣變得可用。如果這些變量中的任何一個(gè)不是由激活的應(yīng)用級(jí)或請(qǐng)求級(jí)context訪問(wèn),會(huì)產(chǎn)生錯(cuò)誤。在后面的章節(jié)會(huì)詳細(xì)討論四個(gè)context變量,所以不要擔(dān)心你不理解它們的用處。
下面的Python shell會(huì)話演示了應(yīng)用級(jí)context是如何工作的:
>>> from hello import app >>> from flask import current_app >>> current_app.name Traceback (most recent call last): ... RuntimeError: working outside of the application context >>> app_ctx = app.app_context() >>> app_ctx.push() >>> current_app.name 'hello' >>> app_ctx.pop()
在這個(gè)示例中,當(dāng)應(yīng)用級(jí)context沒(méi)有激活,但是卻作為有效的context被壓入棧中,current_app.name報(bào)錯(cuò)。注意在應(yīng)用程序?qū)嵗幸粋€(gè)應(yīng)用級(jí)context是如何通過(guò)調(diào)用app.app_context()來(lái)獲得的。
5.2、請(qǐng)求調(diào)度
當(dāng)一個(gè)應(yīng)用程序收到客戶端的請(qǐng)求,它需要找到響應(yīng)的視圖函數(shù)為之服務(wù)。對(duì)于這個(gè)任務(wù),F(xiàn)lask會(huì)在應(yīng)用程序的URL映射中查找請(qǐng)求的URL,該映射包含URLs和操作它們的視圖函數(shù)。Flask通過(guò)app.route裝飾器或非裝飾器版本app.add_url_rule()來(lái)建立這個(gè)映射。
看一下Flask應(yīng)用程序中URL映射是怎樣的,你可以在Python shell中檢查hello.py創(chuàng)建的映射。測(cè)試中,請(qǐng)確保你的虛擬環(huán)境是激活狀態(tài):
(venv) % python >>> from hello import app >>> app.url_map Map([<Rule '/' (HEAD, OPTIONS, GET) -> index>, <Rule '/static/<filename>' (HEAD, OPTIONS, GET) -> static>, <Rule '/user/<name>' (HEAD, OPTIONS, GET) -> user>])
/和/user/<name>路由是由應(yīng)用程序中的app.route所定義。/static/<filename>路由是由Flask添加,用于訪問(wèn)靜態(tài)文件的一個(gè)特殊路由。你將在第三章學(xué)習(xí)更多關(guān)于靜態(tài)文件的內(nèi)容。
URL映射中所示的HEAD、OPTIONS、GET元素為request方法,由路由處理。Flask連接方法到每個(gè)路由,這樣不同的請(qǐng)求方法發(fā)送到相同的URL可以被不同的視圖函數(shù)處理。HEAD和OPTIONS方法由Flask自動(dòng)管理,所以實(shí)際上可以說(shuō),在這個(gè)應(yīng)用程序中URL映射的三個(gè)路由都連接到GET方法了。在第四章你將學(xué)習(xí)為路由指定不同的請(qǐng)求方法。
5.3、請(qǐng)求Hooks
有些時(shí)候在每個(gè)請(qǐng)求處理之前或之后執(zhí)行代碼是非常有用的。例如,在開(kāi)始每一個(gè)請(qǐng)求前可能有必要?jiǎng)?chuàng)建數(shù)據(jù)庫(kù)連接,或?qū)τ脩粽?qǐng)求進(jìn)行驗(yàn)證。為了避免復(fù)制處理這些操作的代碼到每一個(gè)視圖函數(shù)中,F(xiàn)lask給你選擇注冊(cè)相同函數(shù)來(lái)調(diào)用,在請(qǐng)求被分配給視圖函數(shù)之前或之后。
請(qǐng)求hooks由裝飾器實(shí)現(xiàn)。下面是四個(gè)Flask支持的hooks:
(1)before_first_request:在第一個(gè)請(qǐng)求被處理前注冊(cè)一個(gè)函數(shù)運(yùn)行。
(2)before_request:在每一個(gè)請(qǐng)求前注冊(cè)一個(gè)函數(shù)運(yùn)行。
(3)after_request:如果沒(méi)有未處理的異常發(fā)生,在每一個(gè)請(qǐng)求后注冊(cè)一個(gè)函數(shù)運(yùn)行。
(4)teardown_request:即使未處理的異常發(fā)生,在每一個(gè)請(qǐng)求后注冊(cè)一個(gè)函數(shù)運(yùn)行。
在請(qǐng)求hook函數(shù)和視圖函數(shù)之間共享數(shù)據(jù)的慣用方法就是使用g全局context。例如,before_request處理程序可以從數(shù)據(jù)庫(kù)加載已登錄的用戶并保存在g.user中。之后,當(dāng)視圖函數(shù)被調(diào)用,可以從那訪問(wèn)用戶。
請(qǐng)求hooks的示例會(huì)在未來(lái)的章節(jié)中展示給大家,所以不用擔(dān)心,
5.4、響應(yīng)
當(dāng)Flask調(diào)用一個(gè)視圖函數(shù),并期望它的返回值去響應(yīng)該請(qǐng)求。大多數(shù)的響應(yīng)是將簡(jiǎn)單字符串構(gòu)成的HTML頁(yè)面發(fā)回給客戶端。
但是HTTP協(xié)議需要比字符串更多的信息作為請(qǐng)求的響應(yīng)。一個(gè)HTTP響應(yīng)中非常重要的部分是狀態(tài)碼,F(xiàn)lask默認(rèn)設(shè)置200來(lái)指示請(qǐng)求已經(jīng)成功處理。
當(dāng)視圖函數(shù)需要用不同的狀態(tài)碼響應(yīng),可以在響應(yīng)文本后添加數(shù)字碼作為第二個(gè)返回值。例如,下面的視圖函數(shù)返回400錯(cuò)誤狀態(tài)碼的請(qǐng)求:
@app.route('/') def index(): return '<h1>Bad Request</h1>', 400
視圖函數(shù)返回的響應(yīng)還可以攜帶第三個(gè)參數(shù),添加一個(gè)頭部字典給HTTP響應(yīng)。通常很少用到,但是你可以在第十四章看到示例。
除了返回一個(gè)、兩個(gè)或三個(gè)值的元組,F(xiàn)lask視圖函數(shù)可以選擇返回response對(duì)象。make_response()函數(shù)可攜帶一個(gè)、兩個(gè)或三個(gè)參數(shù),和視圖函數(shù)返回的值一樣,并返回一個(gè)response對(duì)象。有時(shí)候在視圖函數(shù)中執(zhí)行這個(gè)轉(zhuǎn)換是非常有用的,然后使用response對(duì)象中的方法進(jìn)一步配置響應(yīng)。下面的示例創(chuàng)建response對(duì)象并設(shè)置cookie:
from flask import make_response @app.route('/') def index(): response = make_response(' <h1>This document carries a cookie!</h1> ') response.set_cookie('answer', '42') return response
有一類特殊的響應(yīng)稱作重定向。這類響應(yīng)不包含頁(yè)面文檔;只是給瀏覽器一個(gè)新的URL去加載新的頁(yè)面。重定向通常和web表單一起使用,你將在第四章學(xué)習(xí)。
重定向通常由302響應(yīng)狀態(tài)碼注明并且重定向的URL由頭部的Location給出。重定向響應(yīng)可以使用三個(gè)值的返回生成,也可通過(guò)響應(yīng)對(duì)象生成,但是鑒于它頻繁的使用,F(xiàn)lask提供redirect()函數(shù)來(lái)創(chuàng)建這樣的響應(yīng):
from flask import redirect @app.route('/') def index(): return redirect('http://www.example.com')
另一個(gè)具有中斷功能的特殊響應(yīng)用來(lái)錯(cuò)誤處理。下面的示例,當(dāng)URL給出的id動(dòng)態(tài)參數(shù)不是一個(gè)合法的用戶時(shí)返回狀態(tài)碼404:
from flask import abort @app.route('/user/<id>') def get_user(id): user = load_user(id) if not user: abort(404) return ' <h1>Hello, %s</h1> ' % user.name
注意終止不是指將控制權(quán)返回給調(diào)用它的函數(shù),而是指通過(guò)拋出異常將控制權(quán)返回給web服務(wù)。
6、Flask擴(kuò)展
Flask是可擴(kuò)展的。它故意騰出地給重要的功能,例如數(shù)據(jù)庫(kù)和用戶授權(quán),給你自由去選擇最適合你的應(yīng)用程序的包,或?qū)懸粋€(gè)自己想要的。
社區(qū)開(kāi)發(fā)了非常多的擴(kuò)展用于各種用途,如果這還不夠,可以使用任何Python標(biāo)準(zhǔn)包和庫(kù)。為了讓你了解一個(gè)擴(kuò)展是如何并入一個(gè)應(yīng)用程序的,下面的章節(jié)給hello.py添加一個(gè)擴(kuò)展,增加應(yīng)用程序的命令行參數(shù)。
6.1、Flask-Script命令行選項(xiàng)
Flask開(kāi)發(fā),其web服務(wù)器支持一系列的啟動(dòng)配置選項(xiàng),但是配置它們的唯一方式只有在腳本中傳遞參數(shù)給app.run()并調(diào)用。這不是非常的方便,理想方法是通過(guò)命令行參數(shù)傳遞配置選項(xiàng)。
Flask-Script是給你的Flask應(yīng)用程序添加命令行解釋的擴(kuò)展。它打包了一組通用的選項(xiàng),還支持自定義命令。
使用pip安裝擴(kuò)展:
(venv) $ pip install flask-script
下面展示了在 hello.py 應(yīng)用程序中添加命令行解釋的變化。
示例. hello.py:使用Flask-Script
from flask.ext.script import Manager manager = Manager(app) # ... if __name__ == '__main__': manager.run()
專為Flask開(kāi)發(fā)的擴(kuò)展暴露在flask.ext命名空間下。Flask-Script從flask.ext.script中導(dǎo)出一個(gè)名為Manager的類。
初始化這個(gè)擴(kuò)展的方法和其他許多擴(kuò)展一樣:主類實(shí)例的初始化是通過(guò)將應(yīng)用程序?qū)嵗鳛閰?shù)傳遞給構(gòu)造函數(shù)實(shí)現(xiàn)的。創(chuàng)建的對(duì)象適當(dāng)?shù)挠糜诿恳粋€(gè)擴(kuò)展。在這個(gè)示例中,服務(wù)器啟動(dòng)通過(guò)manager.run()來(lái)路由,且命令行在這被解析。
建議:如果你有克隆在GitHub上的應(yīng)用程序,你現(xiàn)在可以運(yùn)行g(shù)it checkout 2c來(lái)切換到這個(gè)版本的應(yīng)用程序。
因?yàn)檫@些變化,應(yīng)用程序獲得一組基本的命令行選項(xiàng)。運(yùn)行hello.py顯示可用信息:
$ python hello.py
usage: hello.py [-h] {shell, runserver} ... positional arguments: {shell, runserver} shell 在Flask應(yīng)用程序上下文的內(nèi)部運(yùn)行一個(gè)Python Shell。 runserver 運(yùn)行Flask開(kāi)發(fā)服務(wù)器,例如:app.run() optional arguments: -h, --help 顯示這個(gè)幫助信息并退出
shell命令用于在應(yīng)用程序上下文中啟動(dòng)一個(gè)Python shell會(huì)話。你可以使用這個(gè)會(huì)話去運(yùn)行維護(hù)任務(wù),或測(cè)試,或調(diào)試錯(cuò)誤。
runserver命令,就像它的名稱一樣,啟動(dòng)web服務(wù)。運(yùn)行python hello.py runserver在調(diào)試模式下啟動(dòng)web服務(wù),還有更多的選項(xiàng):
(venv) $ python hello.py runserver --help usage: hello.py runserver [-h] [-t HOST] [-p PORT] [--threaded] [--processes PROCESSES] [--passthrough-errors] [-d] [-r]
運(yùn)行Flask開(kāi)發(fā)服務(wù)器,例如:app.run()
optional arguments: -h, --help 顯示這個(gè)幫助信息并退出 -t HOST, --host HOST -p PORT, --port PORT --threaded --processes PROCESSES --passthrough-errors -d, --no-debug -r, --no-reload
--host參數(shù)是一個(gè)非常有用的選項(xiàng),因?yàn)樗芨嬖Vweb服務(wù)器監(jiān)聽(tīng)哪個(gè)網(wǎng)絡(luò)接口的客戶端連接。默認(rèn),F(xiàn)lask開(kāi)發(fā)的web服務(wù)器監(jiān)聽(tīng)localhost的連接,所以只有來(lái)自內(nèi)部計(jì)算機(jī)運(yùn)行的服務(wù)器可以接收。下面的命令使得web服務(wù)器監(jiān)聽(tīng)公網(wǎng)接口,其他網(wǎng)絡(luò)上的計(jì)算機(jī)可以連接:
(venv) $ python hello.py runserver --host 0.0.0.0 * Running on http://0.0.0.0:5000/ * Restarting with reload
現(xiàn)在web服務(wù)器應(yīng)該可以從網(wǎng)絡(luò)中的任何一臺(tái)計(jì)算機(jī)訪問(wèn) http://a.b.c.d:5000 ,“a.b.c.d”是運(yùn)行服務(wù)的計(jì)算機(jī)的外部IP地址。
相關(guān)文章
python requests包的request()函數(shù)中的參數(shù)-params和data的區(qū)別介紹
這篇文章主要介紹了python requests包的request()函數(shù)中的參數(shù)-params和data的區(qū)別介紹,具有很好參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-05-05python中mechanize庫(kù)的簡(jiǎn)單使用示例
最近的項(xiàng)目中使用到了mechanize庫(kù),下面寫個(gè)簡(jiǎn)單使用的小例子給大家參考2014-01-01Python如何應(yīng)用cx_Oracle獲取oracle中的clob字段問(wèn)題
今天小編就為大家分享一篇Python如何應(yīng)用cx_Oracle獲取oracle中的clob字段問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-08-08Python?httpx庫(kù)終極指南實(shí)戰(zhàn)案例
httpx 由 Encode 團(tuán)隊(duì)開(kāi)發(fā),于 2019 年首次發(fā)布,目標(biāo)是提供一個(gè)現(xiàn)代化的 HTTP 客戶端,支持同步和異步操作,并兼容 HTTP/1.1 和 HTTP/2,這篇文章主要介紹了Python?httpx庫(kù)終極指南,需要的朋友可以參考下2025-05-05基于python實(shí)現(xiàn)鼠標(biāo)實(shí)時(shí)坐標(biāo)監(jiān)測(cè)
這篇文章主要給大家介紹了如何基于python實(shí)現(xiàn)鼠標(biāo)實(shí)時(shí)坐標(biāo)監(jiān)測(cè),文章通過(guò)代碼示例介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下2023-11-11在Python中通過(guò)機(jī)器學(xué)習(xí)實(shí)現(xiàn)人體姿勢(shì)估計(jì)
姿態(tài)檢測(cè)是計(jì)算機(jī)視覺(jué)領(lǐng)域的一個(gè)活躍研究領(lǐng)域。這篇文章將為大家介紹在Python中如何利用機(jī)器學(xué)習(xí)進(jìn)行人體姿勢(shì)估計(jì),感興趣的小伙伴可以了解一下2021-12-12Anaconda3中的Jupyter notebook添加目錄插件的實(shí)現(xiàn)
這篇文章主要介紹了Anaconda3中的Jupyter notebook添加目錄插件的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-05-05