python web框架學(xué)習(xí)筆記
一、web框架本質(zhì)
1.基于socket,自己處理請(qǐng)求
#!/usr/bin/env python3
#coding:utf8
import socket
def handle_request(client):
#接收請(qǐng)求
buf = client.recv(1024)
print(buf)
#返回信息
client.send(bytes('<h1>welcome liuyao webserver</h1>','utf8'))
def main():
#創(chuàng)建sock對(duì)象
sock = socket.socket()
#監(jiān)聽(tīng)80端口
sock.bind(('localhost',8000))
#最大連接數(shù)
sock.listen(5)
print('welcome nginx')
#循環(huán)
while True:
#等待用戶(hù)的連接,默認(rèn)accept阻塞當(dāng)有請(qǐng)求的時(shí)候往下執(zhí)行
connection,address = sock.accept()
#把連接交給handle_request函數(shù)
handle_request(connection)
#關(guān)閉連接
connection.close()
if __name__ == '__main__':
main()
2.基于wsgi
WSGI,全稱(chēng) Web Server Gateway Interface,或者 Python Web Server Gateway Interface ,是為 Python 語(yǔ)言定義的 Web 服務(wù)器和 Web 應(yīng)用程序或框架之間的一種簡(jiǎn)單而通用的接口。自從 WSGI 被開(kāi)發(fā)出來(lái)以后,許多其它語(yǔ)言中也出現(xiàn)了類(lèi)似接口。
WSGI 的官方定義是,the Python Web Server Gateway Interface。從名字就可以看出來(lái),這東西是一個(gè)Gateway,也就是網(wǎng)關(guān)。網(wǎng)關(guān)的作用就是在協(xié)議之間進(jìn)行轉(zhuǎn)換。
WSGI 是作為 Web 服務(wù)器與 Web 應(yīng)用程序或應(yīng)用框架之間的一種低級(jí)別的接口,以提升可移植 Web 應(yīng)用開(kāi)發(fā)的共同點(diǎn)。WSGI 是基于現(xiàn)存的 CGI 標(biāo)準(zhǔn)而設(shè)計(jì)的。
很多框架都自帶了 WSGI server ,比如 Flask,webpy,Django、CherryPy等等。當(dāng)然性能都不好,自帶的 web server 更多的是測(cè)試用途,發(fā)布時(shí)則使用生產(chǎn)環(huán)境的 WSGI server或者是聯(lián)合 nginx 做 uwsgi 。
python標(biāo)準(zhǔn)庫(kù)提供的獨(dú)立WSGI服務(wù)器稱(chēng)為wsgiref。
#!/usr/bin/env python
#coding:utf-8
#導(dǎo)入wsgi模塊
from wsgiref.simple_server import make_server
def RunServer(environ, start_response):
start_response('200 OK', [('Content-Type', 'text/html')])
return [bytes("welcome webserver".encode('utf8'))]
if __name__ == '__main__':
httpd = make_server('', 8000, RunServer)
print ("Serving HTTP on port 8000...")
httpd.serve_forever()
#接收請(qǐng)求
#預(yù)處理請(qǐng)求(封裝了很多http請(qǐng)求的東西)
請(qǐng)求過(guò)來(lái)后就執(zhí)行RunServer這個(gè)函數(shù)。
原理圖:

當(dāng)用戶(hù)發(fā)送請(qǐng)求,socket將請(qǐng)求交給函數(shù)處理,之后再返回給用戶(hù)。
二、自定義web框架
python標(biāo)準(zhǔn)庫(kù)提供的wsgiref模塊開(kāi)發(fā)一個(gè)自己的Web框架
之前的使用wsgiref只能訪(fǎng)問(wèn)一個(gè)url
下面這個(gè)可以根據(jù)你訪(fǎng)問(wèn)的不同url請(qǐng)求進(jìn)行處理并且返回給用戶(hù)
#!/usr/bin/env python
#coding:utf-8
from wsgiref.simple_server import make_server
def RunServer(environ, start_response):
start_response('200 OK', [('Content-Type','text/html')])
#根據(jù)url的不同,返回不同的字符串
#1 獲取URL[URL從哪里獲取?當(dāng)請(qǐng)求過(guò)來(lái)之后執(zhí)行RunServer,
#wsgi給咱們封裝了這些請(qǐng)求,這些請(qǐng)求都封裝到了,environ & start_response]
request_url = environ['PATH_INFO']
print (request_url)
#2 根據(jù)URL做不同的相應(yīng)
#print environ #這里可以通過(guò)斷點(diǎn)來(lái)查看它都封裝了什么數(shù)據(jù)
if request_url == '/login':
return [bytes("welcome login",'utf8')]
elif request_url == '/reg':
return [bytes("welcome reg",'utf8')]
else:
return [bytes('<h1>404! no found</h1>','utf8')]
if __name__ == '__main__':
httpd = make_server('', 8000, RunServer)
print ("Serving HTTP on port 8000...")
httpd.serve_forever()
當(dāng)然 以上雖然根據(jù)不同url來(lái)進(jìn)行處理,但是如果大量url的話(huà),那么代碼寫(xiě)起來(lái)就很繁瑣。
所以使用下面方法進(jìn)行處理
#!/usr/bin/env python
#coding:utf-8
from wsgiref.simple_server import make_server
def index():
return [bytes('<h1>index</h1>','utf8')]
def login():
return [bytes('<h1>login</h1>','utf8')]
def reg():
return [bytes('<h1>reg</h1>','utf8')]
def layout():
return [bytes('<h1>layout</h1>','utf8')]
#定義一個(gè)列表 把url和上面的函數(shù)做一個(gè)對(duì)應(yīng)
urllist = [
('/index',index),
('/login',login),
('/reg',reg),
('/layout',layout),
]
def RunServer(environ, start_response):
start_response('200 OK', [('Content-Type','text/html')])
#根據(jù)url的不同,返回不同的字符串
#1 獲取URL[URL從哪里獲取?當(dāng)請(qǐng)求過(guò)來(lái)之后執(zhí)行RunServer,wsgi給咱們封裝了這些請(qǐng)求,這些請(qǐng)求都封裝到了,environ & start_response]
request_url = environ['PATH_INFO']
print (request_url)
#2 根據(jù)URL做不同的相應(yīng)
#print environ #這里可以通過(guò)斷點(diǎn)來(lái)查看它都封裝了什么數(shù)據(jù)
#循環(huán)這個(gè)列表 找到你打開(kāi)的url 返回url對(duì)應(yīng)的函數(shù)
for url in urllist:
if request_url == url[0]:
return url[1]()
else:
#url_list列表里都沒(méi)有返回404
return [bytes('<h1>404 not found</h1>','utf8')]
if __name__ == '__main__':
httpd = make_server('', 8000, RunServer)
print ("Serving HTTP on port 8000...")
httpd.serve_forever()
三、模板引擎
對(duì)應(yīng)上面的操作 都是根據(jù)用戶(hù)訪(fǎng)問(wèn)的url返回給用戶(hù)一個(gè)字符串的 比如return xxx
案例:
首先寫(xiě)一個(gè)index.html頁(yè)面
內(nèi)容:
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>index</title> </head> <body> <h1>welcome index</h1> </body> </html>
login.html頁(yè)面
內(nèi)容:
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>login</title> </head> <body> <h1>welcome login</h1> <form> user:<input type="text"/> pass:<input type="password"/> <button type="button">login in</button> </form> </body> </html>
python代碼:
#!/usr/bin/env python
#coding:utf-8
from wsgiref.simple_server import make_server
def index():
#把index頁(yè)面讀進(jìn)來(lái)返回給用戶(hù)
indexfile = open('index.html','r+').read()
return [bytes(indexfile,'utf8')]
def login():
loginfile = open('login.html','r+').read()
return [bytes(loginfile,'utf8')]
urllist = [
('/login',login),
('/index',index),
]
def RunServer(environ, start_response):
start_response('200 OK', [('Content-Type','text/html')])
#根據(jù)url的不同,返回不同的字符串
#1 獲取URL[URL從哪里獲取?當(dāng)請(qǐng)求過(guò)來(lái)之后執(zhí)行RunServer,wsgi給咱們封裝了這些請(qǐng)求,這些請(qǐng)求都封裝到了,environ & start_response]
request_url = environ['PATH_INFO']
print (request_url)
#2 根據(jù)URL做不同的相應(yīng)
#print environ #這里可以通過(guò)斷點(diǎn)來(lái)查看它都封裝了什么數(shù)據(jù)
for url in urllist:
#如果用戶(hù)請(qǐng)求的url和咱們定義的rul匹配
if request_url == url[0]:
#執(zhí)行
return url[1]()
else:
#url_list列表里都沒(méi)有返回404
return [bytes('<h1>404 not found</h1>','utf8')]
if __name__ == '__main__':
httpd = make_server('', 8000, RunServer)
print ("Serving HTTP on port 8000...")
httpd.serve_forever()
但是以上內(nèi)容只能返回給靜態(tài)內(nèi)容,不能返回動(dòng)態(tài)內(nèi)容
那么如何返回動(dòng)態(tài)內(nèi)容呢
自定義一套特殊的語(yǔ)法,進(jìn)行替換
使用開(kāi)源工具jinja2,遵循其指定語(yǔ)法
index.html 遵循jinja語(yǔ)法進(jìn)行替換、循環(huán)、判斷
先展示大概效果,具體jinja2會(huì)在下章django筆記來(lái)進(jìn)行詳細(xì)說(shuō)明
index.html頁(yè)面
內(nèi)容:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body>
<!--general replace-->
<h1>{{ name }}</h1>
<h1>{{ age }}</h1>
<h1>{{ time }}</h1>
<!--for circular replace-->
<ul>
{% for item in user_list %}
<li>{{ item }}</li>
{% endfor %}
</ul>
<!--if else judge-->
{% if num == 1 %}
<h1>num == 1</h1>
{% else %}
<h1>num == 2</h1>
{% endif %}
</body>
</html>
python代碼:
#!/usr/bin/env python
#-*- coding:utf-8 -*-
import time
#導(dǎo)入wsgi模塊
from wsgiref.simple_server import make_server
#導(dǎo)入jinja模塊
from jinja2 import Template
def index():
#打開(kāi)index.html
data = open('index.html').read()
#使用jinja2渲染
template = Template(data)
result = template.render(
name = 'yaoyao',
age = '18',
time = str(time.time()),
user_list = ['linux','python','bootstarp'],
num = 1
)
#同樣是替換為什么用jinja,因?yàn)樗粌H僅是文本的他還支持if判斷 & for循環(huán) 操作
#這里需要注意因?yàn)槟J(rèn)是的unicode的編碼所以設(shè)置為utf-8
return [bytes(result,'utf8')]
urllist = [
('/index',index),
]
def RunServer(environ, start_response):
start_response('200 OK', [('Content-Type', 'text/html')])
#根據(jù)url的不同,返回不同的字符串
#1 獲取URL[URL從哪里獲取?當(dāng)請(qǐng)求過(guò)來(lái)之后執(zhí)行RunServer,
# wsgi給咱們封裝了這些請(qǐng)求,這些請(qǐng)求都封裝到了,environ & start_response]
request_url = environ['PATH_INFO']
print(request_url)
#2 根據(jù)URL做不同的相應(yīng)
#循環(huán)這個(gè)列表
for url in urllist:
#如果用戶(hù)請(qǐng)求的url和咱們定義的rul匹配
if request_url == url[0]:
print (url)
return url[1]()
else:
#urllist列表里都沒(méi)有返回404
return [bytes('<h1>404 not found</h1>','utf8')]
if __name__ == '__main__':
httpd = make_server('', 8000, RunServer)
print ("Serving HTTP on port 8000...")
httpd.serve_forever()
四、MVC和MTV
1.MVC
全名是Model View Controller,是模型(model)-視圖(view)-控制器(controller)的縮寫(xiě),一種軟件設(shè)計(jì)典范,用一種業(yè)務(wù)邏輯、數(shù)據(jù)、界面顯示分離的方法組織代碼,將業(yè)務(wù)邏輯聚集到一個(gè)部件里面,在改進(jìn)和個(gè)性化定制界面及用戶(hù)交互的同時(shí),不需要重新編寫(xiě)業(yè)務(wù)邏輯。MVC被獨(dú)特的發(fā)展起來(lái)用于映射傳統(tǒng)的輸入、處理和輸出功能在一個(gè)邏輯的圖形化用戶(hù)界面的結(jié)構(gòu)中。

將路由規(guī)則放入urls.py
操作urls的放入controller里的func函數(shù)
將數(shù)據(jù)庫(kù)操作黨風(fēng)model里的db.py里
將html頁(yè)面等放入views里
原理圖:

2.MTV
Models 處理DB操作
Templates html模板
Views 處理函數(shù)請(qǐng)求

原理圖:

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助。
相關(guān)文章
Python?torch.onnx.export用法詳細(xì)介紹
這篇文章主要給大家介紹了關(guān)于Python?torch.onnx.export用法詳細(xì)介紹的相關(guān)資料,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2022-07-07
python??UPX?is?not?available問(wèn)題解決方法
這篇文章主要介紹了python?UPX?is?not?available問(wèn)題解決,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-04-04
python時(shí)間與Unix時(shí)間戳相互轉(zhuǎn)換方法詳解
這篇文章主要介紹了python時(shí)間與Unix時(shí)間戳相互轉(zhuǎn)換方法詳解,需要的朋友可以參考下2020-02-02
詳談在flask中使用jsonify和json.dumps的區(qū)別
下面小編就為大家分享一篇詳談在flask中使用jsonify和json.dumps的區(qū)別,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-03-03
Python深度學(xué)習(xí)TensorFlow神經(jīng)網(wǎng)絡(luò)基礎(chǔ)概括
這篇文章主要為大家介紹了Python深度學(xué)習(xí)中TensorFlow神經(jīng)網(wǎng)絡(luò)基礎(chǔ)概括,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步2021-10-10
Python中將兩個(gè)或多個(gè)list合成一個(gè)list的方法小結(jié)
python中,list這種數(shù)據(jù)結(jié)構(gòu)很常用到,如果兩個(gè)或者多個(gè)list結(jié)構(gòu)相同,內(nèi)容類(lèi)型相同,我們通常會(huì)將兩個(gè)或者多個(gè)list合并成一個(gè),這樣我們?cè)傺h(huán)遍歷的時(shí)候就可以一次性處理掉了2019-05-05
Python異常對(duì)代碼運(yùn)行性能的影響實(shí)例解析
這篇文章主要介紹了Python異常對(duì)代碼運(yùn)行性能的影響實(shí)例解析,分享了相關(guān)代碼示例,小編覺(jué)得還是挺不錯(cuò)的,具有一定借鑒價(jià)值,需要的朋友可以參考下2018-02-02
Python爬蟲(chóng)實(shí)現(xiàn)selenium處理iframe作用域問(wèn)題
這篇文章主要介紹了Python爬蟲(chóng)實(shí)現(xiàn)selenium處理iframe作用域問(wèn)題,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-01-01

