Tornado之自定义异步非阻塞的服务器和客户端
一、自定义的异步非阻塞的客户端
#!/usr/bin/env python
# -*- coding: utf8 -*-
# __Author: "Skiler Hao"
# date: 2017/5/16 15:04
import select
import socket
import pprint """
自定义了异步IO模块
利用非阻塞的socket,不等待连接是否成功,不等待请求的响应
select模块,去监听创建的套接字,是否有准备写,准备读的 """ class HttpResponse:
def __init__(self, response_data):
self.raw_data = response_data
self.data = str(response_data, encoding='utf-8')
self.Header = {}
self.GET = {}
self.BODY = {}
self.response_info = ''
self.initialize() def initialize(self):
header_data, body_data = self.data.split('\r\n\r\n', 1)
self.BODY = body_data header_list = header_data.split('\r\n')
# print(header_list)
for header_item in header_list:
header_byte = header_item.split(':', 1)
if len(header_byte) == 2:
self.Header[header_byte[0]] = header_byte[1] else:
self.response_info = header_byte def __str__(self):
return self.response_info class HttpRequest:
"""
对HttpRequest的简单封装,将socket,host和callback封装成一个对象
""" def __init__(self, sk, host, callback):
"""
收到socket,host, callback回调函数,将其封装成HttRequest对象
:param sk:一个socket对象
:param host:主机名或者域名
:param callback:socket结束后的回调函数
"""
self.socket = sk
self.host = host
self.callback = callback def fileno(self):
"""
select可以直接监听
定义一个fileno()并返回一个,可以将HttpRequest()对象直接放在select直接去轮训,监听
:return:fileno()文件描述符
"""
return self.socket.fileno() class AsyncHttpClient:
"""
异步Http客户端
""" def __init__(self):
self.conn = []
self.connection = [] def add_request(self, host, callback):
"""
传过来一个host和callback,自动将其封装HttpRequest对象,然后将其加入到self.conn和self.connection中
:param host: 传过来一个host,一个回调函数callback
:param callback:
:return:
"""
sk = socket.socket()
sk.setblocking(0) # 创建一个非阻塞的socket
try:
sk.connect((host, 80)) # 这个socket去连指定的主机的80端口
except BlockingIOError as e:
pass
# 将socket,host,callback封装成一个对象
http_request = HttpRequest(sk, host, callback) # 将对象追加到conn和connection
self.conn.append(http_request)
self.connection.append(http_request) def run(self):
# 任务调用的接口
while True:
# 监听套接字的信号
# select(rlist,wlist,xlist)
# 三个列表,select分别监听 三个列表,rlist是否有读信号,wlist是否写信号,xlist是否有异常信号,最后是timeout
# 一旦有某个对象相应的信号,就将当前的当前的对象返回给对应的位置
rlist, wlist, xlist = select.select(self.conn, self.connection, self.conn, 0.05)
# 一旦有写信号
for w in wlist:
# 首次就是服务器,告知连接成功,你可以往里面写数据啦
# print(w.host,'连接成功,赶紧写数据吧!')
# 在套接字赶紧写上我要获取你的首页
tpl = "GET / HTTP/1.0\r\nHost:%s\r\n\r\n" % (w.host,)
w.socket.send(bytes(tpl, encoding='utf-8'))
# select就不用监听这个套接字是否有写入信号,直接移除掉
self.connection.remove(w) # 一旦有读信号
for r in rlist:
# print(r.host,'开始收到数据啦~~~') rec_data = bytes()
while True:
# 开始收数据,直到接收不到数据
try:
chunk = r.socket.recv(8096) # 一次接受8096个字节的数据
rec_data += chunk
except BlockingIOError as e:
break
# print(r.host,rec_data) # 执行打包中的callback方法,将接收到数据传过去
r.callback(rec_data)
r.socket.close()
# 获取到相应的相应数据,就不在监听其是否有读数据
self.conn.remove(r)
if len(self.conn) == 0:
# 如果没有要监听是否有读信号的套接字,就可以跳出循环任务结束了
break def f1(rec_data):
response = HttpResponse(rec_data)
print(response.BODY) def f2(data):
print('输出到屏幕') # 定义一个字典列表,字典列表包含主机名和回调函数
url_list = [
{'host': 'www.baidu.com', 'callback': f1},
{'host': 'cn.bing.com', 'callback': f1},
{'host': 'www.cnblogs.com', 'callback': f1},
]
# 声明一个异步Async requestclient = AsyncHttpClient() for item in url_list:
requestclient.add_request(item['host'], item['callback']) requestclient.run()
二、自定义的异步非阻塞的服务端
#!/usr/bin/env python
# -*- coding: utf8 -*-
# __Author: "Skiler Hao"
# date: 2017/5/19 7:05
import socket, select EOL1 = b'/r/n'
EOL2 = b'/r/n/r/n' # 拼接成的response
response = b'HTTP/1.0 200 OK/r/nDate: Mon, 1 Jan 1996 01:01:01 GMT/r/n'
response += b'Content-Type: text/plain/r/nContent-Length: 13/r/n/r/n'
response += b'Hello, world!' # 创建一个服务端的socket,来监听是否有请求过来
serversocket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
serversocket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
serversocket.bind(('0.0.0.0', 8080)) # 绑定
serversocket.listen(1) # 监听
serversocket.setblocking(0) # 设置时非阻塞
print(serversocket.fileno())
epoll = select.epoll() # 准备设计一个IO多路复用
epoll.register(serversocket.fileno(), select.EPOLLIN) # 把上面的serversocket的fileno() 和 监听准备读信号 注册到epoll中去 try:
# 设置三个空字典
connections = {}
requests = {}
responses = {}
while True:
# 查看epoll是否有信号有的话,就放在events中
events = epoll.poll(1) # 循环events,分别拿到 文件描述号 和对应的事件
for fileno, event in events:
# 如果当前的文件描述号是serversocket,那么说明有新的连接
if fileno == serversocket.fileno():
# 所以就得接受,创建了 连接,拿到了对方的IP地址
connection, address = serversocket.accept()
# connection就是客户端连接过来建立的socket,设置为非阻塞
connection.setblocking(0)
# 客户端建立的socket也注册到select模块的IO多路复用中去
epoll.register(connection.fileno(), select.EPOLLIN) # 以Connection的文件描述号 作为键 socket作为值保存在connections中
connections[connection.fileno()] = connection # 同时在requests和responses字典中,
# requests中 以connection.fileno() 作为键 以请求的内容作为值
# responses中 以connection.fileno() 作为键 以相应的内容作为值,这个我们返回的是固定的,仅仅返回hello world requests[connection.fileno()] = b''
responses[connection.fileno()] = response # 如果请求的数据不是socketserver,那肯定是客户端的,判断是否是准备读的信号
elif event & select.EPOLLIN:
# 立马来开始读取数据,加到requests对象套接字的内容中去
requests[fileno] += connections[fileno].recv(1024) # 判断换行 和 两个换行是否在接收过来的数据中
if EOL1 in requests[fileno] or EOL2 in requests[fileno]:
# 如果是的话,就将这个套接字的监听更新为准备写
epoll.modify(fileno, select.EPOLLOUT) # 打印40个-,然后换行,加上请求的内容
print('-' * 40 + '/n' + requests[fileno].decode()[:-2])
# 如果请求的数据不是socketserver,那肯定是客户端的,判断是否是准备写的信号
elif event & select.EPOLLOUT:
# 立马来开始读取数据,就将response的对应的套接字号对应的值拿出来,其实就是hello world,O(∩_∩)O哈哈~
# 并统计发送了多少个字节
byteswritten = connections[fileno].send(responses[fileno])
# 更新response对应套接字内容为剩下的内容
responses[fileno] = responses[fileno][byteswritten:]
# 如果内容发完了,剩下的长度就是0,如果长度是0
if len(responses[fileno]) == 0:
# 就修改select,不监听该套接字的内容,其就变成了EPOLLHUP
epoll.modify(fileno, 0)
# 然后关闭该socket的管道
connections[fileno].shutdown(socket.SHUT_RDWR)
elif event & select.EPOLLHUP:
# 如果不监听该套接字的内容,就将其注销掉
epoll.unregister(fileno)
# 关闭该套接字
connections[fileno].close()
# 从连接中删除该文件描述符
del connections[fileno]
finally:
# 最后关闭serversocket服务器套接字
epoll.unregister(serversocket.fileno())
# 关闭epoll
epoll.close()
# 套接字关闭
serversocket.close()
Tornado之自定义异步非阻塞的服务器和客户端的更多相关文章
- 03: 自定义异步非阻塞tornado框架
目录:Tornado其他篇 01: tornado基础篇 02: tornado进阶篇 03: 自定义异步非阻塞tornado框架 04: 打开tornado源码剖析处理过程 目录: 1.1 源码 1 ...
- 200行自定义异步非阻塞Web框架
Python的Web框架中Tornado以异步非阻塞而闻名.本篇将使用200行代码完成一个微型异步非阻塞Web框架:Snow. 一.源码 本文基于非阻塞的Socket以及IO多路复用从而实现异步非阻塞 ...
- Tornado异步非阻塞的使用以及原理
Tornado 和现在的主流 Web 服务器框架(包括大多数 Python 的框架)有着明显的区别:它是非阻塞式服务器,而且速度相当快.得利于其 非阻塞的方式和对 epoll 的运用,Tornado ...
- Tornado之异步非阻塞
同步模式:同步模式下,只有处理完前一个任务下一个才会执行 class MainHandler(tornado.web.RequestHandler): def get(self): time.slee ...
- NIO:异步非阻塞I/O,AIO,BIO
Neety的基础使用及说明 https://www.cnblogs.com/rrong/p/9712847.html BIO(缺乏弹性伸缩能力,并发量小,容易出现内存溢出,出现宕机 每一个客户端对应一 ...
- 使用tornado让你的请求异步非阻塞
http://www.dongwm.com/archives/shi-yong-tornadorang-ni-de-qing-qiu-yi-bu-fei-zu-sai/?utm_source=tuic ...
- 异步非阻塞IO的Python Web框架--Tornado
Tornado的全称是Torado Web Server,从名字上就可知它可用作Web服务器,但同时它也是一个Python Web的开发框架.最初是在FriendFeed公司的网站上使用,FaceBo ...
- Tornado的异步非阻塞
阻塞和非阻塞Web框架 只有Tornado和Node.js是异步非阻塞的,其他所有的web框架都是阻塞式的. Tornado阻塞和非阻塞两种模式都支持. 阻塞式: 代表:Django.Flask.To ...
- Tornado 异步非阻塞
1 装饰器 + Future 从而实现Tornado的异步非阻塞 class AsyncHandler(tornado.web.RequestHandler): @gen.coroutine def ...
随机推荐
- yii2高级模板安装
通过 Composer 安装 如果还没有安装 Composer,在 Linux 和 Mac OS X 中可以运行如下命令: curl -sS https://getcomposer.org/insta ...
- 转:kafka入门
一.基本概念 介绍 Kafka是一个分布式的.可分区的.可复制的消息系统.它提供了普通消息系统的功能,但具有自己独特的设计. 这个独特的设计是什么样的呢? 首先让我们看几个基本的消息系统术语:Kafk ...
- 由浅入深了解EventBus:(五)
事件分发 EventBus3.0的事件的分发时通过EventBus类中的post(粘性事件为postSticky)方法,post与postSticky的唯一区别就是,在postSticky内部首先会向 ...
- SQL 二进制和字符互转
1.二进制转为字符串 ALTER function varbin2hexstr( ) )) as begin ),@i int select @re='',@i=datalength(@bin) ), ...
- Conky配置文件
Conky是一个可以在linux系统中实时显示系统性能的工具,美观且十分好用,我们选择安装conky-all程序包 # set to yes if you want Conky to be forke ...
- APUE学习笔记——4.2结构体 struct stat 及其相关函数介绍
以下不少内容来自man手册 结构体struct stat 结构体struct stat用于保存文件相关的所有信息. struct stat的基本成员如下所示 struc ...
- JSP和JS的区别
从本科毕业设计开始就一直困扰我,jsp和js这两者的区别,一直处于迷糊状态,也没有搞清楚.今天就简单的介绍下两者的区别. 1.JSP全称是java server page JS全称是javaSc ...
- d3.js(v5.7)树状图
一.新建画布 二.数据处理 三.绘制连接线 图示: 四.绘制节点.文字 图示: 五.总结 path元素:其实就是定义了绘图的坐标点,从哪开始,移动到哪,怎样移动(命令) 具体可百度(或许以后我会总结一 ...
- online learning,batch learning&批量梯度下降,随机梯度下降
以上几个概念之前没有完全弄清其含义及区别,容易混淆概念,在本文浅析一下: 一.online learning vs batch learning online learning强调的是学习是实时的,流 ...
- Vue学习笔记 ——v-html
v-html: 在网页中,后台传来的json数据中包含html标签,将该json数据绑定到Vue.js中对象中,对该对象进行for循环,发现数据中的html标签不能被解析,而是当作字符显示出来 解决: ...