前言: 今天寫一篇博客簡單介紹下web框架,下篇博客介紹下django框架~~
一、WSGI
眾所周知,對於所有的Web應用,本質上其實就是一個socket服務端,用戶的瀏覽器其實就是一個socket客戶端。

1 #!/usr/bin/env python 2 #coding:utf-8 3 4 import socket 5 6 def handle_request(client): 7 buf = client.recv(1024) 8 client.send("HTTP/1.1 200 OK\r\n\r\n") 9 client.send("Hello, Seven") 10 11 def main(): 12 sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) 13 sock.bind(('localhost',8000)) 14 sock.listen(5) 15 16 while True: 17 connection, address = sock.accept() 18 handle_request(connection) 19 connection.close() 20 21 if __name__ == '__main__': 22 main()
對於真實開發中的python web程序來說,一般會分為兩部分:服務器程序和應用程序。服務器程序負責對socket服務器進行封裝,並在請求到來時,對請求的各種數據進行整理。應用程序則負責具體的邏輯處理。為了方便應用程序的開發,就出現了眾多的Web框架,例如:Django、Flask、web.py 等。不同的框架有不同的開發方式,但是無論如何,開發出的應用程序都要和服務器程序配合,才能為用戶提供服務。這樣,服務器程序就需要為不同的框架提供不同的支持。這樣混亂的局面無論對於服務器還是框架,都是不好的。對服務器來說,需要支持各種不同框架,對框架來說,只有支持它的服務器才能被開發出的應用使用。這時候,標准化就變得尤為重要。我們可以設立一個標准,只要服務器程序支持這個標准,框架也支持這個標准,那么他們就可以配合使用。一旦標准確定,雙方各自實現。這樣,服務器可以支持更多支持標准的框架,框架也可以使用更多支持標准的服務器。
WSGI(Web Server Gateway Interface)WEB服務網關接口是一種規范,它定義了使用python編寫的web app與web server之間接口格式,實現web app與web server間的解耦。
python標准庫提供的獨立WSGI服務器稱為wsgiref。
二、兩種web框架
WEB框架分為兩種,第一種不用WSGI標准,如Tornado框架,需要自己寫socket;第二種則用到了WSGI封裝好的socket標准,如Django/bottle/flask/web.py框架。
三、web小實例
接下用python的wsgiref模塊做個小實例,該程序是在python2.7環境運行的,在python3.4下出現莫名錯誤~~
1 #coding:utf-8
2 from wsgiref.simple_server import make_server 3 4 def f1(): 5 return "index"
6 7 def f2(): 8 return "news"
9 10 routers = { 11 "/index/": f1, 12 "/news/": f2, 13 } 14 15 # environ包含用戶請求的所有信息,是wsgi封裝好的
16 def RunServer(environ, start_response): 17 start_response('200 OK', [('Content-Type', 'text/html')]) 18 url = environ["PATH_INFO"] 19 """
20 if url == "/index/": 21 return "index" 22 elif url == "/news/": 23 return "news" 24 else: 25 return "404" 26 """
27 if url in routers.keys(): 28 func_name = routers[url] 29 return func_name() 30 else: 31 return "404"
32 33 34 if __name__ == '__main__': 35 # 創建socket server對象(參數可查看源碼構造函數,如果找不到可往父類找直至找到)
36 httpd = make_server('', 8000, RunServer) 37 print("Serving HTTP on port 8000...") 38 # 相當while循環,等待用戶請求到來
39 # 只要有請求進來,執行RunServer函數
40 httpd.serve_forever()
運行結果:
四、模板引擎
在上一步驟中,對於所有的index、news均返回給用戶瀏覽器一個簡單的字符串,在現實的Web請求中一般會返回一個復雜的符合HTML規則的字符串,所以我們一般將要返回給用戶的HTML寫在指定文件中,然后再返回。
雖然可以返回給用戶HTML的內容以現實復雜的頁面,但是還是存在問題:如何給用戶返回動態內容?
- 自定義一套特殊的語法,進行替換
- 使用開源工具jinja2,遵循其指定語法
首先你要先導入jinja2模塊,注意是在python2.x環境下導入。
導入成功后,先來看看下面的例子,了解下jinja2的用法:
目錄結構:
templates/t1.html
1 <!DOCTYPE html>
2 <html lang="en">
3 <head>
4 <meta charset="UTF-8">
5 <title>Title</title>
6 </head>
7 <body>
8 <h1>zcl</h1>
9 <div>
10 <a style="color: red;">blog</a>
11 <div style="font-size: large;">zcl ---- ((x))</div>
12 </div>
13 </body>
14 </html>
templates/t2.html:
注意jinja2替換的語法規則:有循環則用{% ... %}, 替換字符則用{{...}}.
1 <!DOCTYPE html>
2 <html lang="en">
3 <head>
4 <meta charset="UTF-8">
5 <title>Title</title>
6 </head>
7 <body>
8 <h1>{{name}}</h1>
9 <hr />
10 <ul>
11 {% for item in user_list %} 12 <li>{{item}}</li> 13 {% endfor %} 14 </ul>
15
16 </body>
17 </html>
web_1.py:
1 #coding:utf-8
2 from wsgiref.simple_server import make_server 3 import time 4 from jinja2 import Template 5
6
7 def f1(): 8 f = open("templates/t1.html") 9 data = f.read() 10 f.close() 11 db_str = str(time.time()) 12 data = data.replace("((x))", db_str) #將t1.html中的((x))值替換成db_str即當前時間 13 return data 14
15
16 def f2(): 17 f = open('templates/t2.html') 18 result = f.read() 19 template = Template(result) 20 # 接收值,進行特殊的替換
21 data = template.render(name='John Doe', user_list=['alex', 'eric']) 22 return data.encode('utf-8') #必須加encode('utf-8')否則無法訪問網頁 23 routers = { 24 "/index/": f1, 25 "/news/": f2, 26 } 27
28 # environ包含用戶請求的所有信息,是wsgi封裝好的
29 def RunServer(environ, start_response): 30 start_response('200 OK', [('Content-Type', 'text/html')]) 31 url = environ["PATH_INFO"] 32
33 if url in routers.keys(): 34 func_name = routers[url] 35 return func_name() 36 else: 37 return "404"
38
39
40 if __name__ == '__main__': 41 # 創建socket server對象(參數可查看源碼構造函數,如果找不到可往父類找直至找到)
42 httpd = make_server('', 8000, RunServer) 43 print("Serving HTTP on port 8000...") 44 # while循環,等待用戶請求到來
45 # 只要有請求進來,執行RunServer函數
46 httpd.serve_forever()
訪問http://127.0.0.1:8000/news/與http://127.0.0.1:8000/index/
五、MVC
前陣子在網上找實習有看到要求熟悉MVC。什么是MVC呢?
這個圖我可是畫了很久的!!我們已經知道服務端返回HTML字符串給客戶端(瀏覽器),但並不是單單從HTML文檔取數據再直接通過路由系統-->WSGI發送給客戶端。比如我博客園的個人主頁,當我點擊https://home.cnblogs.com/u/0zcl/時,會看到此時我只有25個粉絲,但25這個數據肯定不是在HTML文檔的吧!在哪里?當然是在數據庫中啦。所以當你點擊我的個人主頁,其實此時不僅要從HTML文檔取數據,還要從數據庫中取數據,再通過jinja2的替換規則進行替換(模板渲染)。你懂了么??
好,還是說回MVC吧。其實MVC只是代碼的目錄結構,只是聽起來很高大上而已~~
想必各位一定知道模塊化吧(找實習也有看到這個要求哈)。根據上圖,我們可以把處理數據(庫)的功能模塊放在一個目錄下,存放HTML的模塊放在一個目錄下:
- M(models): 數據庫處理
- V(views): 存放HTML模板
- C(controllers): 處理用戶請求
MTV與MVC同理:
MVC目錄例子:
django框架是基於MTV模式的,下面我創建一個django程序(注意看目錄結構):