Web Socket

所谓 Web 服务,本质上就是用户使用一个 socket 客户端(浏览器)去访问一个 socket 服务端。

下面是一个最基础的基于 socket 的 Python Web 服务端程序。

import socket

# 最简单的web程序
def handle_request(connection):
content = connection.recv(1024)
print(content)
connection.send(bytes("HTTP/1.1 200 OK\r\n\r\n".encode("utf-8")))
connection.send(bytes("hello, World!".encode("utf-8"))) def service():
server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
server_address = ('127.0.0.1', 8000)
server.bind(server_address)
server.listen(5) while True:
connection, client_address = server.accept()
print(connection)
handle_request(connection)
connection.close() if __name__ == '__main__':
service()

WSGI (Web Server Gateway Interface)

上面的例子展示了 Web 的本质。

但是对于现实中的 Python Web 程序来说,一般服务端程序会分为两部分:

  • 服务器程序(用来接收、整理客户端发送的请求)
  • 应用程序(处理服务器程序传递过来的请求)

而在服务端开发应用程序时,我们会把常用的功能函数封装起来,这就是各种 Web 框架了。

Python 常见的 Web 框架有 Django,Flask,Tornado 等等。

不同的框架提供了不同程度和方式的封装,方便程序员在其上进行二次开发。

但是,服务器程序应用程序 需要相互配合才能给用户提供服务,因此我们需要制定一个标准,只要双方同时遵守这个标准,就可以相互配合了。

在 Python 中,这个标准就是 WSGI 了。

WSGI 是一种规范,它规定了服务器程序和应用程序各自使用的接口和功能,实现了二者的解耦。

Python 标准库提供的独立 WSGI 服务器称之为 wsgiref

from wsgiref.simple_server import make_server

def application(environ, start_response):
start_response('200 OK', [('Content-Type', 'text/html')])
return [bytes('<h1>Hello, World</h1>'.encode("utf-8")), b'42'] def run_server():
server = make_server("127.0.0.1", 8000, application)
server.serve_forever() if __name__ == '__main__':
run_server()

路径处理

上面的例子虽然实现了一个正常的服务端功能,但却有一个巨大的缺陷,那就是不同的 url 也只能返回相同的内容,为了解决这个问题,我们对代码作出如下修改。

from wsgiref.simple_server import make_server

def application(environ, start_response):
url = environ['PATH_INFO']
print("url:", url)
start_response('200 OK', [('Content-Type', 'text/html')])
if url == "/index":
return [bytes('<h1>index!</h1>'.encode("utf-8")), b'42']
elif url == "/login":
return [bytes('<h1>login !</h1>'.encode("utf-8")), b'42']
elif url == "/logout":
return [bytes('<h1>logout !</h1>'.encode("utf-8")), b'42']
else:
return [bytes('<h1>404 !</h1>'.encode("utf-8")), b'42'] def run_server():
server = make_server("127.0.0.1", 8000, application)
server.serve_forever() if __name__ == '__main__':
run_server()

通过从 environ 中获取存储着路径的变量 PATH_INFO。我们就可以对不同的 url 返回不同的结果了。

分层处理

上面的例子虽然做到了对不同的路径返回不同的结果,但是所有的逻辑都写在一个函数内,一旦体量增加就会混乱不堪。

因此我们需要对应用程序进行进一步的解耦。

from wsgiref.simple_server import make_server

def index():
return [bytes('<h1>index!</h1>'.encode("utf-8")), b'42'] def login():
return [bytes('<h1>login !</h1>'.encode("utf-8")), b'42'] def logout():
return [bytes('<h1>logout !</h1>'.encode("utf-8")), b'42'] urlConf = [
("/index", index),
("/login", login),
("/logout", logout),
] # 不同的网址有不同的结果,但是所有的处理逻辑写到一起,很混乱
def application(environ, start_response):
url = environ['PATH_INFO']
print("url:", url) response_fun = None for item in urlConf:
if url == item[0]:
response_fun = item[1]
break if response_fun:
start_response('200 OK', [('Content-Type', 'text/html')])
response_body = response_fun()
else:
start_response('404 Not Found', [('Content-Type', 'text/html')])
response_body = [bytes('<h1>404 !</h1>'.encode("utf-8")), b'42']
return response_body def run_server():
server = make_server("127.0.0.1", 8000, application)
server.serve_forever() if __name__ == '__main__':
run_server()



怎么样是不是已经有点像我们平时使用的各种框架了。

urlConf用户根据不同的路径,配置不同的函数。

server程序中遍历urlConf来根据路径定位所要执行的函数。

文件拆分

上面的例子已经有了框架的雏形了,但所有的功能都在一个 Python 文件中,很不优雅。

这时你就可以将代码分成三个文件。

view.py:专门用户存放各种页面的处理函数

url.py 配置路径和函数的关系

server.py 执行web的主程序

将不同功能的代码分门别类存放,进行进一步的解耦。

Python Web 框架原理的更多相关文章

  1. Django之web框架原理

    Web框架原理 我们可以这样理解:所有的Web应用本质上就是一个socket服务端,而用户的浏览器就是一个socket客户端. 这样我们就可以自己实现Web框架了. 先写一个 原始的web框架 imp ...

  2. Werkzeug——python web开发工具包

    转载请注明原文地址:https://www.cnblogs.com/ygj0930/p/10826062.html 一:Werkzeug是个啥 1)Werkzeug是一个工具包,它封装了很多东西,诸如 ...

  3. Django安装和web框架原理

    Django安装和web框架原理 在PyCharm中安装 在cmd中输入 django-admin startproject 项目名 如果报错 不是内部或外部命令,也不是可运行的程序 需要添加环境变量 ...

  4. Redis的Python实践,以及四中常用应用场景详解——学习董伟明老师的《Python Web开发实践》

    首先,简单介绍:Redis是一个基于内存的键值对存储系统,常用作数据库.缓存和消息代理. 支持:字符串,字典,列表,集合,有序集合,位图(bitmaps),地理位置,HyperLogLog等多种数据结 ...

  5. Python Web 方向(一)

    Python Web 方向(一) --------Django站点创建 文章地址:http://www.cnblogs.com/likeli/p/5821744.html Python版本:2.7 推 ...

  6. 浅谈五大Python Web框架

    转载:http://feilong.me/2011/01/talk-about-Python-web-framework 说到Web Framework,Ruby的世界Rails一统江湖,而Pytho ...

  7. nginx上部署python web

    nginx上部署python web http://uwsgi-docs.readthedocs.io/en/latest/tutorials/Django_and_nginx.html

  8. 全面解读python web 程序的9种部署方式

    转载自鲁塔弗的博客,本文地址http://lutaf.com/141.htm  python有很多web 开发框架,代码写完了,部署上线是个大事,通常来说,web应用一般是三层结构 web serve ...

  9. Python Web 开发的十个框架【转载】

    Python 是一门动态.面向对象语言.其最初就是作为一门面向对象语言设计的,并且在后期又加入了一些更高级的特性.除了语言本身的设计目的之外,Python标准 库也是值得大家称赞的,Python甚至还 ...

随机推荐

  1. CUP监测1分钟(60s)的python的matplotlib动态作图

    import matplotlib.pyplot as plt import psutil import time # 第1步,导出通用字体设置 from matplotlib import font ...

  2. LoadLibrary与LoadLibraryEx

    参考了这个博客:https://blog.csdn.net/cuglifangzheng/article/details/50580279 以及这个博客:https://blog.csdn.net/z ...

  3. AppBar 自定义顶部导航按钮 图标、颜色 以及 TabBar 定义顶部 Tab 切换

    一.Flutter AppBar 自定义顶部按钮图标.颜色 leading   在标题前面显示的一个控件,在首页通常显示应用的 logo:在其他界面通常显示为返回按钮 title  标题,通常显示为当 ...

  4. Mac安装php扩展redis遇到的问题,执行phpize问题

    1.安装redis在mac OS中可以使用brew命令进行安装redis:mac OS使用brew命令安装软件安装命令:brew install redis因为我已经安装过了,这里就不在赘述.安装完之 ...

  5. 广度优先搜索(BFS)与深度优先搜索(DFS)的对比及优缺点

    深搜,顾名思义,是深入其中.直取结果的一种搜索方法. 如果深搜是一个人,那么他的性格一定倔得像头牛!他从一点出发去旅游,只朝着一个方向走,除非路断了,他绝不改变方向!除非四个方向全都不通或遇到终点,他 ...

  6. win10中,vscode安装go插件排雷指南

    最近学习go,想着使用强大的vscode编写go,在安装go插件过程中,遇到了很多问题.下面记录解决方案. 1)win10环境,安装go,vscode,git 配置GOPATH环境变量,在我的电脑-& ...

  7. 「JSOI2013」哈利波特和死亡圣器

    「JSOI2013」哈利波特和死亡圣器 传送门 首先二分,这没什么好说的. 然后就成了一个恒成立问题,就是说我们需要满足最坏情况下的需求. 那么显然在最坏情况下伏地魔是不会走回头路的 因为这显然是白给 ...

  8. Javascript——(1)

    1.Javascript有两种解释表示形式:1)在html的<header>中写<script><script/>,另一种是将另一个文件保存为xxx.js文档,然后 ...

  9. ThinkPHP3.2.2的函数扩展

    ThinkPHP的函数扩展:为了更好的在前台模板中显示变量,例如,商品分类中,分类名称之间的缩进.此时,在APP/Common/Common文件夹下(APP为新建的应用目录),新建一个php文件,如: ...

  10. JS bind()方法、JS原生实现bind()

    一.arguments的含义 // arguments 是一个对应于传递给函数的参数的类数组对象 function a(){ console.log(arguments); } a(); // Arg ...