1. 程式人生 > >伺服器程式和應用程式

伺服器程式和應用程式

對於真實開發中的python web程式來說,一般會分為兩部分:伺服器程式和應用程式。

伺服器程式負責對socket服務端進行封裝,並在請求到來時,對請求的各種資料進行整理。

應用程式則負責具體的邏輯處理。為了方便應用程式的開發,就出現了眾多的web框架,例如:Django、Flask、web.py等。不同的框架有不同的開發方式,但是如論如何,開發出的應用程式都要和伺服器程式配合,才能為使用者提供服務。

這樣,伺服器程式就需要為不同的框架提供不同的支援,這種混亂的局面無論對於伺服器還是框架,都是不好的。對於伺服器來說,需要支援不同的框架;對框架來說,只有支援它的伺服器才能別開發出的應用使用。

這時候標準化就變得非常重要,我們可以設定一個標準,只要伺服器程式支援這個標準,框架也支援這個標準,那麼他們就可以配合使用。一旦標準確定,雙方各自實現。這樣,伺服器就可以支援更多支援標準的框架,框架也可以使用更多支援標準的伺服器。

WSGI(Web Server Gateway Interface)就是一種規範,它定義了使用Python編寫的web應用程式與web伺服器程式之間的介面格式,顯示web應用程式與web伺服器程式間的解耦。

常用的WSGI伺服器有uwsgi,Gunicom。而Python標準庫提供的獨立WSGI伺服器叫wsgiref, Django開發環境用的就是這個模組來做伺服器。

1. wsgiref

使用wsgiref模組來替換自己寫的web框剪的socket server部分:

 1 from wsgiref.simple_server import make_server   
 2 #將返回不同的內容部分封裝成函式   
 3 def index(url):   
 4     # 讀取index.html頁面的內容   
 5     with open("index.html", "r", encoding="utf8") as f:   
 6         s = f.read()   
 7     # 返回位元組資料   
8 return bytes(s, encoding="utf8") 9 10 11 def home(url): 12 with open("home.html", "r", encoding="utf8") as f: 13 s = f.read() 14 return bytes(s, encoding="utf8") 15 16 17 def timer(url): 18 import time 19 with open("time.html", "r", encoding="utf8") as f: 20 s = f.read() 21 s = s.replace('@@[email protected]@', time.strftime("%Y-%m-%d %H:%M:%S")) 22 return bytes(s, encoding="utf8") 23 24 25 # 定義一個url和實際要執行的函式的對應關係 26 list1 = [ 27 ("/index/", index), 28 ("/home/", home), 29 ("/time/", timer), 30 ] 31 32 33 def run_server(environ, start_response): 34 start_response('200 OK', [('Content-Type', 'text/html;charset=utf8'), ]) # 設定HTTP響應的狀態碼和頭資訊 35 url = environ['PATH_INFO'] # 取到使用者輸入的url 36 func = None 37 for i in list1: 38 if i[0] == url: 39 func = i[1] 40 break 41 if func: 42 response = func(url) 43 else: 44 response = b"404 not found!" 45 return [response, ] 46 47 48 if __name__ == '__main__': 49 httpd = make_server('127.0.0.1', 8090, run_server) 50 httpd.serve_forever()
wsgiref模組版

 2. jinja2

上面的程式碼只是實現了一個簡單的動態,完全可以從資料庫中查詢資料,然後去替換html中對應的內容,然後再發送給瀏覽器來完成渲染。

這個過程就相當於HTML模板渲染資料。

本質上就是HTML內容中利用一些特殊的符號來替換要展示的資料。

下載jinja2的命令:

pip3 install jinja2

2.1 使用jinja2渲染index.html檔案

 首先建立一個名稱為index.html的檔案,程式碼如下:

 1 <!DOCTYPE html>
 2 <html lang="zh-CN">
 3 <head>
 4   <meta charset="UTF-8">
 5   <meta http-equiv="x-ua-compatible" content="IE=edge">
 6   <meta name="viewport" content="width=device-width, initial-scale=1">
 7   <title>Title</title>
 8 </head>
 9 <body>
10     <h1>姓名:{{name}}</h1>
11     <h1>愛好:</h1>
12     <ul>
13         {% for hobby in hobby_list %}
14         <li>{{hobby}}</li>
15         {% endfor %}
16     </ul>
17 </body>
18 </html>
index.html

使用jinja2渲染index.html檔案程式碼如下:

 1 from wsgiref.simple_server import make_server  
 2 from jinja2 import Template  
 3   
 4   
 5 def index(url):  
 6     # 讀取HTML檔案內容  
 7     with open("index2.html", "r", encoding="utf8") as f:  
 8         data = f.read()  
 9         template = Template(data)   # 生成模板檔案  
10         ret = template.render({'name': 'yang', 'hobby_list': ['讀書', '運動', '寫程式碼']})   # 把資料填充到模板中  
11     return bytes(ret, encoding="utf8")  
12   
13   
14 def home(url):  
15     with open("home.html", "r", encoding="utf8") as f:  
16         s = f.read()  
17     return bytes(s, encoding="utf8")  
18   
19   
20 # 定義一個url和實際要執行的函式的對應關係  
21 list1 = [  
22     ("/index/", index),  
23     ("/home/", home),  
24 ]  
25   
26   
27 def run_server(environ, start_response):  
28     start_response('200 OK', [('Content-Type', 'text/html;charset=utf8'), ])  # 設定HTTP響應的狀態碼和頭資訊  
29     url = environ['PATH_INFO']  # 取到使用者輸入的url  
30     func = None  
31     for i in list1:  
32         if i[0] == url:  
33             func = i[1]  
34             break  
35     if func:  
36         response = func(url)  
37     else:  
38         response = b"404 not found!"  
39     return [response, ]  
40   
41   
42 if __name__ == '__main__':  
43     httpd = make_server('127.0.0.1', 8090, run_server)  
44     print("我在8090等你哦...")  
45     httpd.serve_forever()
jinja2渲染index.html

現在頁面中的資料都是我們自己手寫的,但是還可以從資料庫查詢資料,來填充頁面。
使用pymysql連線資料庫:

1 conn = pymysql.connect(host="127.0.0.1", port=3306, user="root", passwd="xxx", db="xxx", charset="utf8")
2 cursor = conn.cursor(cursor=pymysql.cursors.DictCursor)
3 cursor.execute("select name, age, department_id from userinfo")
4 user_list = cursor.fetchall()
5 cursor.close()
6 conn.close()
pymysql連線資料庫

建立一個測試的user表:

1 CREATE TABLE user(
2   id int auto_increment PRIMARY KEY,
3   name CHAR(10) NOT NULL,
4   hobby CHAR(20) NOT NULL
5 )engine=innodb DEFAULT charset=UTF8;
user表

模板的原理就是字串替換,只要在HTML頁面中遵循jinja2的語法規則編寫,其內部就會按照指定的語法進行相應的替換,從而達到動態的返回內容。