WSGI-mini-web框架服务器
前期准备:
安装python环境
安装pycharm
安装MySQL数据库
安装pymsql
创建一个学生表,存入数据
我们只是实现一个非常简单的web服务,前端页面不会专门做页面文件,会在代码中以具体命令的形式形成文件样式。服务器的功能:
1、show.html显示student表中的数据,点击页面内的修改按钮,会跳转到change.html页面修改内容
2、change.html修改student表中数据,将学生姓名由“张三”改为“李四”,点击页面内的"show page"会重新跳转到显示学生信息页面重新显示修改后的学生信息
创建表
create table student(
id int primary key,
sname varchar(50) not null,
sex char(5) not null
);
原始表数据

一、创建tcp服务器tcpWebServer.py
服务器需要实现多线程的访问,采用面向对象的思想,创建一个服务器类。采用wsgi的思想,需要在服务器类中预先设置一个set_response方法,供框架代码中application的调用
Tcp服务器类创建流程如下:
因为多线程中都需要使用到套接字创建、ip和port的绑定,以及监听(listen),所以我们设计类的时候将这几部分设置为对象属性,直接在初始化方法中创建
创建__init__方法
- 创建套接字绑定ip和port:使用命令bind()
- Listen使套接字变为可以被动链接:使用命令listen()
创建服务器处理请求/发送请求方法
- 接收客户端传送的request数据
- 因为前端发送的数据是url地址所以需要处理url地址的代码
- 设置伪静态的url地址
- 发送response返回给前端
创建set_response函数
创建运行函数
import multiprocessing
import re
import socket
from 简单web服务实现 import web_frame class MSGIServer:
def __init__(self):
"""初始化方法,完成套接字创建、ip和port绑定、listen监听"""
# 创建套接字
self.tcp_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
self.tcp_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) # 绑定ip和端口,使用方法bind((ip, 端口)),注意bind方法需要入参一个元组
# 自己编写的代码不要使用1024以内的端口号
self.tcp_socket.bind(('', 7890)) #监听
self.tcp_socket.listen(128) def client_server(self, new_socket):
"""处理和客户端之间数据的交互:接收recv/发送send"""
# 接收客户端发送过来的请求
# 使用方法recv(),数字参数是指定长度
# decode()方法实现解码,参数为编码格式,因为为使用的window系统,所以使用gbk格式
request = new_socket.recv(1024).decode('gbk')
print(str(request).splitlines()[0])
# 使用正则提取请求url地址的后缀名
# group()表示分组匹配
file_name = re.match('.*/.*\.([a-z]*)\sH', str(request).splitlines()[0]).group(1)
# 使用正则提取请求的url地址,用于后面代码中进行匹配
page_name = re.match('.*/(.*)\sH', str(request).splitlines()[0]).group(1)
# 返回数据给客户端
# 根据请求的url地址的不同,返回不同的数据,伪静态(.html)内容是目标内容,否则一律返回404
if not file_name.endswith('html'):
response = "HTTP/1.1 200 OK \r\n"
response += '\r\n'
response += "<h1>404 NOT FOUND</h1>"
new_socket.send(response.encode('gbk'))
else:
# 创建字典,后面调用application函数作为参数传递
env = dict()
env['path'] = page_name
print('--->page_name=%s<---' % page_name)
# 调用框架模块中的application方法,传入url信息,确定执行的命令
body = web_frame.application(env, self.set_response)
# print('--->body=%s<---' % body)
# response = "HTTP/1.1 200 OK \r\n"
header = 'HTTP/1.1 %s\r\n' % self.status
for temp in self.header:
header += '%s:%s' % (temp[0], temp[1])
header += '\r\n'
response = header + '\r\n' + body new_socket.send(response.encode('utf-8'))
new_socket.close() # 设置请求头
def set_response(self, status, headers):
self.status = status
self.header = headers def run_forver(self):
"""实现web服务器"""
while True:
# 4、等待新客户端的链接 accept
new_socket, socket_address = self.tcp_socket.accept()
# 5、为新客户端服务,独立为服务的方法处理接收和发送数据(request\response)
p = multiprocessing.Process(target=self.client_server, args=(new_socket,))
# client_server(new_socket)
p.start()
new_socket.close()
self.tcp_socket.close() def main():
msgi_server = MSGIServer()
msgi_server.run_forver() if __name__ == '__main__':
main()
二、创建框架web_frame.py
- 创建闭包
- 创建两个函数show(显示学生信息)、change(修改学生信息)
- 创建application函数
import time
import pymysql
import logging URL_FUNC_DICT = dict()
func_list = list() def route(url):
"""创建路由闭包:
实现url地址的路由,即根据装饰器填入的url信息执行装饰器所装饰的方法
"""
def set_func(func):
URL_FUNC_DICT[url] = func def call_func():
func()
return call_func
return set_func @route('show.html')
def show():
"""登录网页"""
sql = "select * from student;"
# 链接MySQL服务器
conn = pymysql.connect(host='127.0.0.1', user='wang', password='', database='jingdong', port=3306, charset='utf8')
cur = conn.cursor()
cur.execute(sql)
# 获取查询到的数据
data = cur.fetchone()
print(data)
cur.close()
conn.close()
# 构建一个网页内容
table = """<html>
<head>
<meta charset="utf-8">
<title>学生信息</title>
</head>
<body>
<tr>
<td>%s</td>
<td>%s</td>
<td>%s</td>
<td>
<a href="/change.html"><input type="button" value="修改"></a>
</td>
</tr>
</body>
</html>"""
# 将数据库查询到的数据拼接到网页内容中
res = table % (str(data[0]), data[1], data[2])
# return cur.fetchone()[0]+str
return res @route('change.html')
def change():
"""修改数据页面"""
# 链接MySQL服务器
sql = "update student set sname='李四';"
# 创建数据库链接对象
conn = pymysql.connect(host='127.0.0.1', user='wang', password='', database='jingdong', port=3306,
charset='utf8')
# 创建游标
cur = conn.cursor()
# 执行sql命令
lines = cur.execute(sql)
# 提交数据
conn.commit()
cur.close()
conn.close()
return "<a href=\"show.html\">show page</a>" # 服务器代码调用执行的代码
# 遵照wsgi协议,框架中需要存在一个application(字典, 函数引用)函数
# 字典,用来传入前端所提交的信息 引用的函数时服务器创建的封装响应头的函数
def application(environ, start_response):
# 调用tcpWebServer模块中的设置相应函数,设置响应头
start_response('200 OK', [('Context-Type', 'text/html')])
# 获取前端传入的字典中的路径,即url地址
page_name = environ['path']
# 设置日志
logging.basicConfig(level=logging.INFO,
filename='./log.txt',
filemode='a',
# format表示日志文件中显示的格式
# astime时间 写入文件名 第几行的日志信息 日志等级 打印的信息
format='%(asctime)s-%(filename)s[line:%(lineno)d]-%(levelname)s:%(message)s'
)
logging.info('访问的是:%s' % page_name)
if page_name in URL_FUNC_DICT.keys():
return URL_FUNC_DICT[page_name]()
else:
return 'not Found'
运行效果
代码运行,浏览器输入"localhost:7890/show.html"查看学生的信息

点击“修改”按钮会跳转到change.html页面

再点击“show page”链接会重新跳转回show.html页面,显示修改后的学生信息

数据库中的数据也已经同步更新

代码目录下已经生成了日志文件

内容为我们设定的显示内容

以上是一个非常粗糙的web服务,后续我们会使用到Django框架来详细设计一个完整的web服务
WSGI-mini-web框架服务器的更多相关文章
- [Python之路] 实现简易HTTP服务器与MINI WEB框架(利用WSGI实现服务器与框架解耦)
本文描述如果简单实现自定义Web服务器与自定义简易框架,并且不断进行版本迭代,从而清晰的展现服务器与Web框架之间是如何结合.如何配合工作的.以及WSGI是什么. 本文帖的代码有点多,但基本每次迭代修 ...
- Wsgi的web框架实例
建立server服务端: from wsgiref.simple_server import make_server import time def f1(request): return [b'&l ...
- [Python之路] 使用装饰器给Web框架添加路由功能(静态、动态、伪静态URL)
一.观察以下代码 以下来自 Python实现简易HTTP服务器与MINI WEB框架(利用WSGI实现服务器与框架解耦) 中的mini_frame最后版本的代码: import time def in ...
- python学习之路web框架
WEB框架的本质 python的WEB框架分为两大类: 1.自己写socket,自己处理请求 2.基于wsgi(Web Server Gateway Interface WEB服务网关接口),自己处理 ...
- 07 返回多个页面web框架
07 返回多个页面web框架 服务器server端python程序(不同页面版本): import socket server=socket.socket() server.bind(("1 ...
- 06 返回静态文件的映射(函数/多线程)web框架
06 返回静态文件的映射(函数/多线程)web框架 服务器server端python程序(函数版): import socket server = socket.socket() server.bin ...
- 05 返回静态文件的多线程web框架
05 返回静态文件的多线程web框架 服务器server端python程序(多线程版): import socket from threading import Thread,currentThrea ...
- 04 返回静态文件的函数web框架
04 返回静态文件的函数web框架 服务器server端python程序(函数版): import socket server = socket.socket() server.bind((" ...
- 03 返回静态文件的高级web框架
03 返回静态文件的高级web框架 服务器server端python程序(高级版): import socket server=socket.socket() server.bind(("1 ...
- 02 初级版web框架
02 初级版web框架 服务器server端python程序(初级版): import socket server=socket.socket() server.bind(("127.0.0 ...
随机推荐
- Windows可以往外ping,外部却ping不通本机
网络背景:192.168.1.17.192.168.1.19.192.168.1.20 三台机器都在一个exsi宿主机下网络非常简单 问题描述:17和19都可以ping通20:20却不能ping通17 ...
- Spring @Column的注解详解
就像@Table注解用来标识实体类与数据表的对应关系类似,@Column注解来标识实体类中属性与数据表中字段的对应关系. 该注解的定义如下: @Target({METHOD, FIELD}) @Ret ...
- dhcp server
centos yum install dhcp -y cat /etc/dhcp/dhcpd.conf default-lease-time 7200; max-lease-time 14400; s ...
- Docker Dockerfile基本配置
1.dockerfile介绍 Dockerfile是Docker用来构建镜像的文本文件,包含自定义的指令和格式.可以通过docker build命令从Dockerfile中构建镜像.这个过程与传统分布 ...
- verilog求倒数-ROM实现方法
采用线性逼近法结合32段线性查找表的方式来实现1/z的计算. 首先将1/32-1/64的定点化数据存放到ROM中,ROM中存放的是扩大了2^20 次方的数字四舍五入后的整数部分.n值越大,精度越大,误 ...
- HDU-1425-sort(计数排序以及快速排序和堆排序的变种)
计数排序 Accepted 1425 483MS 5276K 997 B G++ #include "bits/stdc++.h" using namespace std; typ ...
- 关于angular跳转路由之后不能自动回到顶部的解决方法
Question: angular2 scroll top on router change 当我们在第一个路由滑动到底部当我们点击导航跳转到另一个路由时页面没有回到顶部而是保持上一个路由的滚动位置, ...
- [洛谷P3403] 跳楼机
题目传送门 套路题,同余最短路. 先只考虑y.z进行连边,再在mod x的意义下进行计算. 这里的“距离”dis[i]指的是,在所有满足a mod x=i的a里,能到达的最小的a是多少. 显然只要能到 ...
- HTTP接口抓包工具之Fiddler
Fiddler的基本功能介绍: Fiddler是最强大最好用的Web调试工具之一,它能记录所有客户端和服务器的http和https请求,允许你监视,设置断点,甚至修改输入输出数据,Fiddler包含了 ...
- 改了改之前那个很糙的XXX
将就着用X度去爬吧 <?php echo "***************************************\r\n"; echo "* SubDom ...