標籤:sof env 模板引擎 ;; 處理 router 返回 httpd 模板
一、web架構本質
眾所周知,對於所有的Web應用,本質上其實就是一個socket服務端,使用者的瀏覽器其實就是一個socket用戶端。
#!/usr/bin/env python#coding:utf-8 import socket def handle_request(client): buf = client.recv(1024) client.send("HTTP/1.1 200 OK\r\n\r\n") client.send("Hello, Seven") def main(): sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) sock.bind((‘localhost‘,8000)) sock.listen(5) while True: connection, address = sock.accept() handle_request(connection) connection.close() if __name__ == ‘__main__‘: main()
上述通過socket來實現了其本質;
而對於真實開發中的python web程式來說,一般會分為兩部分:伺服器程式和應用程式。
伺服器程式負責對socket伺服器進行封裝,並在請求到來時,對請求的各種資料進行整理。
應用程式則負責具體的邏輯處理。
為了方便應用程式的開發,就出現了眾多的Web架構,例如:Django、Flask、web.py 等。不同的架構有不同的開發方式,但是無論如何,開發出的應用程式都要和伺服器程式配合,才能為使用者提供服務。這樣,伺服器程式就需要為不同的架構提供不同的支援。這樣混亂的局面無論對於伺服器還是架構,都是不好的。對伺服器來說,需要支援各種不同架構,對架構來說,只有支援它的伺服器才能被開發出的應用使用。這時候,標準化就變得尤為重要。我們可以設立一個標準,只要伺服器程式支援這個標準,架構也支援這個標準,那麼他們就可以配合使用。一旦標準確定,雙方各自實現。這樣,伺服器可以支援更多支援標準的架構,架構也可以使用更多支援標準的伺服器。
WSGI(Web Server Gateway Interface)是一種規範,它定義了使用python編寫的web app與web server之間介面格式,實現web app與web server間的解耦。
python標準庫提供的獨立WSGI伺服器稱為wsgiref。
#!/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‘)]) return ‘<h1>Hello, web!</h1>‘ if __name__ == ‘__main__‘: httpd = make_server(‘‘, 8000, RunServer) print "Serving HTTP on port 8000..." httpd.serve_forever()
二、自訂web架構
1. 架構
通過python標準庫提供的wsgiref模組開發一個自己的Web架構
#!/usr/bin/env python#coding:utf-8from wsgiref.simple_server import make_server def index(): return ‘index‘ def login(): return ‘login‘ def routers(): urlpatterns = ( (‘/index/‘,index), (‘/login/‘,login), ) return urlpatterns def RunServer(environ, start_response): start_response(‘200 OK‘, [(‘Content-Type‘, ‘text/html‘)]) url = environ[‘PATH_INFO‘] urlpatterns = routers() func = None for item in urlpatterns: if item[0] == url: func = item[1] break if func: return func() else: return ‘404 not found‘ if __name__ == ‘__main__‘: httpd = make_server(‘‘, 8000, RunServer) print "Serving HTTP on port 8000..." httpd.serve_forever()
2. 模板引擎
在上一步驟中,對於所有的login、index均返回給使用者瀏覽器一個簡單的字串,在現實的Web請求中一般會返回一個複雜的符合HTML規則的字串,所以我們一般將要返回給使用者的HTML寫在指定檔案中,然後再返回。如:
<!DOCTYPE html><html><head lang="en"> <meta charset="UTF-8"> <title></title></head><body> <h1>Index</h1></body></html>index.html
index.html
<!DOCTYPE html><html><head lang="en"> <meta charset="UTF-8"> <title></title></head><body> <form> <input type="text" /> <input type="text" /> <input type="submit" /> </form></body></html>login.html
login.html
#!/usr/bin/env python# -*- coding:utf-8 -*- from wsgiref.simple_server import make_server def index(): # return ‘index‘ f = open(‘index.html‘) data = f.read() return data def login(): # return ‘login‘ f = open(‘login.html‘) data = f.read() return data def routers(): urlpatterns = ( (‘/index/‘, index), (‘/login/‘, login), ) return urlpatterns def run_server(environ, start_response): start_response(‘200 OK‘, [(‘Content-Type‘, ‘text/html‘)]) url = environ[‘PATH_INFO‘] urlpatterns = routers() func = None for item in urlpatterns: if item[0] == url: func = item[1] break if func: return func() else: return ‘404 not found‘ if __name__ == ‘__main__‘: httpd = make_server(‘‘, 8000, run_server) print "Serving HTTP on port 8000..." httpd.serve_forever()
對於上述代碼,雖然可以返回給使用者HTML的內容以現實複雜的頁面,但是還是存在問題:如何給使用者返回動態內容?
3. 返回動態內容
自訂一套特殊的文法,進行替換
使用開源工具jinja2,遵循其指定文法
<!DOCTYPE html><html><head lang="en"> <meta charset="UTF-8"> <title></title></head><body> <h1>{{name}}</h1> <ul> {% for item in user_list %} <li>{{item}}</li> {% endfor %} </ul></body></html>index.htmlindex.html
#!/usr/bin/env python# -*- coding:utf-8 -*- from wsgiref.simple_server import make_serverfrom jinja2 import Template def index(): # return ‘index‘ # template = Template(‘Hello {{ name }}!‘) # result = template.render(name=‘John Doe‘) f = open(‘index.html‘) result = f.read() template = Template(result) data = template.render(name=‘John Doe‘, user_list=[‘alex‘, ‘eric‘]) return data.encode(‘utf-8‘) def login(): # return ‘login‘ f = open(‘login.html‘) data = f.read() return data def routers(): urlpatterns = ( (‘/index/‘, index), (‘/login/‘, login), ) return urlpatterns def run_server(environ, start_response): start_response(‘200 OK‘, [(‘Content-Type‘, ‘text/html‘)]) url = environ[‘PATH_INFO‘] urlpatterns = routers() func = None for item in urlpatterns: if item[0] == url: func = item[1] break if func: return func() else: return ‘404 not found‘ if __name__ == ‘__main__‘: httpd = make_server(‘‘, 8000, run_server) print "Serving HTTP on port 8000..." httpd.serve_forever()
三、常用web架構
MVC Model View Controller 資料庫 模板檔案 業務處理MTV Model Template View 資料庫 模板檔案 業務處理
web前端基礎知識-(六)web架構