一、web框架介紹html
Web框架(Web framework)是一種開發框架,用來支持動態網站、網絡應用和網絡服務的開發。這大多數的web框架提供了一套開發和部署網站的方式,也爲web行爲提供了一套通用的方法。web框架已經實現了不少功能,開發人員使用框架提供的方法而且完成本身的業務邏輯,就能快速開發web應用了。瀏覽器和服務器的是基於HTTP協議進行通訊的。也能夠說web框架就是在以上十幾行代碼基礎張擴展出來的,有不少簡單方便使用的方法,大大提升了開發的效率。前端
二、實現簡單的web框架mysql
基於socket寫一個web應用web
初版本 sql
經過HTTP協議傳送過來的信息返回頁面數據庫
import socket server = socket.socket() server.bind(('127.0.0.1',8000)) server.listen(5) while True: conn,addr = server.accept() while True: data = conn.recv(1024) ''' ***data*** b'GET /index/ HTTP/1.1\r\n Host: 127.0.0.1:8000\r\n Connection: keep-alive\r\n Upgrade-Insecure-Requests: 1\r\n User-Agent: Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/74.0.3729.131 Safari/537.36\r\n Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3\r\n Accept-Encoding: gzip, deflate, br\r\n Accept-Language: zh-CN,zh;q=0.9,und;q=0.8,en;q=0.7\r\n Cookie: csrftoken=kzAhOkqkoPdnQuOxI7AQTa8aOmT7g6ebPiwrI0jpQ8m04NmYLkzkFvDtD8febu41; Hm_lvt_080836300300be57b7f34f4b3e97d911=1559726342,1559732417\r\n\r\n' ''' conn.send(b'HTTP/1.1 200 OK\r\n\r\n') ''' 頁面顯示Hello 必須遵照http協議返回一個響應頭 ''' current_path = data.decode('utf-8').split('\r\n')[0].split(' ')[1] ''' /index/ ''' if current_path == '/index/': with open('index.html', 'rb') as f: conn.send(f.read()) else: conn.send(b'404') conn.close()
第二版本django
根據HTTP傳送過來的信息返回相應的頁面,經過列表保存信息模擬django中的urls,經過函數返回數據模擬django中views瀏覽器
import socket server = socket.socket() server.bind(('127.0.0.1', 8000)) server.listen(5) def index(): return 'index' def login(): return 'login' def error(): return '404' urls = [ ('/index/', index),
('/login/', login) ] while True: conn, addr = server.accept() while True: data = conn.recv(1024) ''' ***data*** b'GET /index/ HTTP/1.1\r\n Host: 127.0.0.1:8000\r\n Connection: keep-alive\r\n Upgrade-Insecure-Requests: 1\r\n User-Agent: Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/74.0.3729.131 Safari/537.36\r\n Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3\r\n Accept-Encoding: gzip, deflate, br\r\n Accept-Language: zh-CN,zh;q=0.9,und;q=0.8,en;q=0.7\r\n Cookie: csrftoken=kzAhOkqkoPdnQuOxI7AQTa8aOmT7g6ebPiwrI0jpQ8m04NmYLkzkFvDtD8febu41; Hm_lvt_080836300300be57b7f34f4b3e97d911=1559726342,1559732417\r\n\r\n' ''' conn.send(b'HTTP/1.1 200 OK\r\n\r\n') ''' 頁面顯示Hello 必須遵照http協議返回一個響應頭 ''' current_path = data.decode('utf-8').split('\r\n')[0].split(' ')[1] ''' /index/ '''
res = None for url in urls: if url[0] == current_path: res = url[1]() break else: res = error() conn.send(res.encode('utf-8'))
第三版服務器
將其拆分紅start.py urls.py views.py網絡
import socket import urls server = socket.socket() server.bind(('127.0.0.1', 8000)) server.listen(5) def error(): return '404' while True: conn, addr = server.accept() while True: data = conn.recv(1024) ''' ***data*** b'GET /index/ HTTP/1.1\r\n Host: 127.0.0.1:8000\r\n Connection: keep-alive\r\n Upgrade-Insecure-Requests: 1\r\n User-Agent: Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/74.0.3729.131 Safari/537.36\r\n Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3\r\n Accept-Encoding: gzip, deflate, br\r\n Accept-Language: zh-CN,zh;q=0.9,und;q=0.8,en;q=0.7\r\n Cookie: csrftoken=kzAhOkqkoPdnQuOxI7AQTa8aOmT7g6ebPiwrI0jpQ8m04NmYLkzkFvDtD8febu41; Hm_lvt_080836300300be57b7f34f4b3e97d911=1559726342,1559732417\r\n\r\n' ''' conn.send(b'HTTP/1.1 200 OK\r\n\r\n') ''' 頁面顯示Hello 必須遵照http協議返回一個響應頭 ''' current_path = data.decode('utf-8').split('\r\n')[0].split(' ')[1] ''' /index/ ''' res = None for url in urls.urls: if url[0] == current_path: res = url[1]() break else: res = error() conn.send(res.encode('utf-8'))
import views urls = [ ('/index/', views.index), ('/login/', views.login) ]
def index(): return 'index' def login(): return 'login'
高級版
使用wsgiref.simple_server的make_server配置web應用
from wsgiref.simple_server import make_server from urls import * def run(env, response): ''' :param env: 請求相關的信息,一個處理好的字符串 :param response: 響應相關信息 :return: ''' response('200 OK', [('username', 'jason'), ('password', '123')]) # 固定寫法 後面列表裏面一個個元祖會以響應頭kv鍵值對的形式返回給客戶端 current_path = env.get('PATH_INFO') func = None for url in urls: if current_path == url[0]: func = url[1] break if func: res = func(env) else: res = error(env) return [res.encode('utf-8')] if __name__ == '__main__': server = make_server('127.0.0.1', 8080, run) server.serve_forever()
def index(env): return 'index' def login(env): return 'login' def error(env): return '404'
from views import * urls = [ ('/index/', index), ('/login/', login) ]
終極版
使用jinja2對模板進行替換,對數據庫進行鏈接
from wsgiref.simple_server import make_server from urls import * def run(env, response): ''' :param env: 請求相關的信息,一個處理好的字符串 :param response: 響應相關信息 :return: ''' response('200 OK', [('username', 'jason'), ('password', '123')]) # 固定寫法 後面列表裏面一個個元祖會以響應頭kv鍵值對的形式返回給客戶端 current_path = env.get('PATH_INFO') func = None for url in urls: if current_path == url[0]: func = url[1] break if func: res = func(env) else: res = error(env) return [res.encode('utf-8')] if __name__ == '__main__': server = make_server('127.0.0.1', 8080, run) server.serve_forever()
import time from jinja2 import Template import pymysql def index(env): return 'index' def login(env): return 'login' def get_time(env): # 先獲取當前時間 current_time = time.strftime('%Y-%m-%d %X') # 打開html文件讀取內容返回給客戶端 with open(r'templates/get_time.html', 'r', encoding='utf-8') as f: data = f.read() # 由於是以r模式打開的文件,全部獲取到的內容就是一堆字符串 res = data.replace('@@time@@', current_time) # 字符串的替換 return res def get_user(env): with open(r'templates/get_user.html', 'r', encoding='utf-8') as f: data = f.read() tmp = Template(data) # 將字典傳遞給前端頁面 前端經過變量名user_dic就能夠獲取到該字典 return tmp.render(user_dic={'name': "jason", 'password': '123'}) def get_db(env): # 鏈接數據庫 獲取數據 渲染到前端頁面 conn = pymysql.connect( host='127.0.0.1', port=3306, user='root', password='123', database='blog', charset='utf8', autocommit=True ) cursor = conn.cursor(pymysql.cursors.DictCursor) cursor.execute('select * from userinfo') user_dict = cursor.fetchall() # [{},{},{},{}] with open(r'templates/get_db.html', 'r', encoding='utf-8') as f: data = f.read() tmp = Template(data) return tmp.render(user_dict=user_dict) def error(env): return '404 error'
from views import * urls = [ ('/index',index), ('/login',login), ('/get_time',get_time), ('/get_user',get_user), ('/get_db',get_db), ]
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> </head> <body> @@time@@ </body> </html>
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> </head> <body> <p>{{ user_dic }}</p> <p>{{ user_dic.name }}</p> <p>{{ user_dic['password'] }}</p> <p>{{ user_dic.get('name') }}</p> </body> </html>