Python之基于socket和select模块实现IO多路复用
'''IO指的是输入输出,一部分指的是文件操作,还有一部分
网络传输操作,例如soekct就是其中之一;多路复用指的是
利用一种机制,同时使用多个IO,例如同时监听多个文件句
柄(socket对象一旦传送或者接收信息),一旦文件句柄出
现变化就会立刻感知到
'''
1、下面通过IO多路复用实现多人同时连接socket服务器 这是服务端代码
import socket sk1 = socket.socket()#sk1,sk2,sk3这就是一个文件描述符
sk1.bind(('127.0.0.1',8002))
sk1.listen() sk2 = socket.socket()
sk2.bind(('127.0.0.1',8003))
sk2.listen() sk3 = socket.socket()
sk3.bind(('127.0.0.1',8004))
sk3.listen() inputs = [sk1,sk2,sk3]
import select
while True:
'''[sk1,sk2,sk3],select内部会自动监听sk1,sk21,sk3三个对象
一旦某个句柄发生变化就会被监听到
'''
#如果有人链接sk1,则会被添加进列表,r_list = [sk1]
r_list,w_list,e_list = select.select(inputs,[],[],1)#1表示等一秒,在while执行到这里的时候监测一秒,没有人来链接的话就接着循环
#print(r_list,sk1.listen())
for sk in r_list:
conn,address = sk.accept()
#print(conn,address)
conn.sendall(bytes('你好',encoding='utf-8'))
conn.close()
客户端的代码都是一样的,就差个端口
#客户端一
import socket obj = socket.socket()
obj.connect(('127.0.0.1',8002))
content = str(obj.recv(1024),encoding='utf-8')
print(content) obj.close() #客户端二
import socket obj = socket.socket()
obj.connect(('127.0.0.1',8003))
content = str(obj.recv(1024),encoding='utf-8')
print(content) obj.close() #客户端三
import socket obj = socket.socket()
obj.connect(('127.0.0.1',8004))
content = str(obj.recv(1024),encoding='utf-8')
print(content) obj.close()
执行结果:只要启动服务器端,然后不同的客户端多次启动都能收到信息,多个端口成功被监听
2、下面使用select模块实现多路复用,使同一个端口同时接收多个链接
import socket sk1 = socket.socket()#sk1,sk2,sk3这就是一个文件描述符
sk1.bind(('127.0.0.1',8002))
sk1.listen() #
# sk2 = socket.socket()
# sk2.bind(('127.0.0.1',8003))
# sk2.listen()
#
# sk3 = socket.socket()
# sk3.bind(('127.0.0.1',8004))
# sk3.listen() inputs = [sk1]
import select #epoll效率更高,但是Windows不支持,它是谁有问题就告诉它,不用循坏
while True:
'''[sk1,sk2,sk3],select内部会自动监听sk1,sk21,sk3三个对象
一旦某个句柄发生变化(某人来链接)就会被监听到
'''
#如果有人链接sk1,则会被添加进列表,r_list = [sk1]
r_list,w_list,e_list = select.select(inputs,[],[],1)
'''第三个参数是监听错误的,只要有错误出现,就会被监听到,返回e_list
第二个参数返回给w_list,只要传了什么,就原封不动的传给w_list'''
print('正在监听 %s 多少个对象' % len(inputs))
for sk in r_list:
if sk == sk1:
#句柄跟服务器端的对象一样,表示有新用户来链接了
conn,address = sk.accept()
inputs.append(conn)#加入去之后,inputs有一个链接对象和服务器对象
else:
#有老用户发消息了
try:
data_byte =sk.recv(1024) data_str =str(data_byte,encoding='utf-8')
sk.sendall(bytes(data_str+'收到了',encoding='utf-8'))
except:#空信息表示客户端断开链接,所以要在监听中移除 inputs.remove(sk)#这里的sk就是之前传进去的conn,因为r_list接收的是有变化的值
启动这个服务端之后,就可以实现多路复用了,可以接收多个客户端同时连接
3、下面介绍一些多路操作里面的读写分离
import socket sk1 = socket.socket()#sk1就是一个文件描述符
sk1.bind(('127.0.0.1',8002))
sk1.listen()
inputs = [sk1]#被检测发生变动的句柄放这里
outputs=[]#用来记录谁给服务端发过信息,以便回复
message_dict = {}#接收信息的,根据句柄形成键值对
import select #epoll效率更高,但是Windows不支持,它是谁有问题就告诉它,不用循坏
while True:
'''[sk1,sk2,sk3],select内部会自动监听sk1,sk21,sk3三个对象
一旦某个句柄发生变化(某人来链接)就会被监听到
'''
#如果有人链接sk1,则会被添加进列表,r_list = [sk1]
r_list,w_list,e_list = select.select(inputs,outputs,inputs,1)
'''第三个参数是监听错误的,只要有错误出现,就会被监听到,返回e_list
第二个参数返回给w_list,只要传了什么,就原封不动的传给w_list'''
print('正在监听 %s 多少个对象' % len(inputs))
for sk in r_list:
if sk == sk1:
#句柄跟服务器端的对象一样,表示有新用户来链接了
conn,address = sk.accept()
inputs.append(conn)#加入去之后,inputs有一个链接对象和服务器对象
message_dict[conn]=[]#字典的key是具体的链接对象
else:
#有老用户发消息了
try:
data_byte =sk.recv(1024) except Exception as e:#空信息表示客户端断开链接,所以要在监听中移除
print(e)
inputs.remove(sk)#这里的sk就是之前传进去的conn,因为r_list接收的是有变化的值
else:
data_str = str(data_byte, encoding='utf-8') message_dict[sk].append(data_str)
outputs.append(sk)#把传送数据的句柄添加进去第二个参数
for conn in w_list:
recv_str=message_dict[conn][0]#拿到我们收到了的信息
del message_dict[conn][0]#删除信息,防止下次出现一样的消息
print(recv_str)
conn.sendall(bytes(recv_str+'收到了',encoding='utf-8'))
outputs.remove(conn)#回复完成之后在列表删除
for sk_or_conn in e_list:
e_list.remove(sk_or_conn)
这样可以形成简单的读写分离操作
对于select里面的参数关系,这里有个武sir画的图

Python之基于socket和select模块实现IO多路复用的更多相关文章
- 在python中编写socket服务端模块(二):使用poll或epoll
在linux上编写socket服务端程序一般可以用select.poll.epoll三种方式,本文主要介绍使用poll和epoll编写socket服务端模块. 使用poll方式的服务器端程序代码: i ...
- C#中级-从零打造基于Socket在线升级模块
一.前言 前段时间一直在折腾基于Socket的产品在线升级模块.之前我曾写过基于.Net Remoting的.基于WCF的在线升级功能,由于并发量较小及当时代码经验的不足一直没有实际应用. ...
- 【Socket】从零打造基于Socket在线升级模块
一.前言 前段时间一直在折腾基于Socket的产品在线升级模块.之前我曾写过基于.Net Remoting的.基于WCF的在线升级功能,由于并发量较小及当时代码经验的不足一直没有实际应用. ...
- select模块(I/O多路复用)
0709自我总结 select模块 一.介绍 Python中的select模块专注于I/O多路复用,提供了select poll epoll三个方法(其中后两个在Linux中可用,windows仅支持 ...
- Python进程、线程、协程及IO多路复用
详情戳击下方链接 Python之进程.线程.协程 python之IO多路复用
- python笔记8 socket(TCP) subprocess模块 粘包现象 struct模块 基于UDP的套接字协议
socket 基于tcp协议socket 服务端 import socket phone = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 买 ...
- Python编写基于socket的非阻塞多人聊天室程序(单线程&多线程)
前置知识:socket非阻塞函数(socket.setblocking(False)),简单多线程编程 代码示例: 1. 单线程非阻塞版本: 服务端: #!/usr/bin/env python # ...
- Python练习-基于socket的FTPServer
# 编辑者:闫龙 import socket,json,struct class MySocket: with open("FtpServiceConfig","r&qu ...
- python selectors模块实现 IO多路复用机制的上传下载
import selectorsimport socketimport os,time BASE_DIR = os.path.dirname(os.path.abspath(__file__))''' ...
随机推荐
- 访问FTP站点下载文件,提示“当前的安全设置不允许从该位置下载文件”的解决方案
访问FTP站点下载文件,提示“当前的安全设置不允许从该位置下载文件”的解决方案: 打开客戶端浏览器--工具---internet-安全-自定义级别-选择到低到中低. 然后点受信任站点,把你要访问的站点 ...
- java Vamei快速教程10 接口的继承和抽象类
作者:Vamei 出处:http://www.cnblogs.com/vamei 欢迎转载,也请保留这段声明.谢谢! 在实施接口中,我们利用interface语法,将interface从类定义中独立出 ...
- linux 命令——5 rm(转)
昨天学习了创建文件和目录的命令mkdir ,今天学习一下linux中删除文件和目录的命令: rm命令.rm是常用的命令,该命令的功能为删除一个目录中的一个或多个文件或目录,它也可以将某个目录及其下的所 ...
- JS.match方法 正则表达式
match() 方法可在字符串内检索指定的值,或找到一个或多个正则表达式的匹配. 该方法类似 indexOf() 和 lastIndexOf(),但是它返回指定的值,而不是字符串的位置. <sc ...
- [OS] 可执行文件的装载
http://www.jianshu.com/p/e1300e7a4c48 1. 虚拟内存 在早期的计算机中,程序是直接运行在物理内存上的,程序在运行时访问的地址就是物理地址.可是,当计算机中同时运行 ...
- BZOJ 1229: [USACO2008 Nov]toy 玩具
BZOJ 1229: [USACO2008 Nov]toy 玩具 标签(空格分隔): OI-BZOJ OI-三分 OI-双端队列 OI-贪心 Time Limit: 10 Sec Memory Lim ...
- Hystrix + Hystrix Dashboard搭建(Spring Cloud 2.X)
本机IP为 192.168.1.102 一.搭建Hystrix Dashboard 1. 新建 Maven 项目 hystrix-dashboard 2. pom.xml <projec ...
- 高性能可扩展MySQL数据库设计及架构优化 电商项目(慕课)第3章 MySQL执行计划(explain)分析
ID:相同就从上而下,不同数字越大越优先
- Everything Be True-freecodecamp算法题目
Everything Be True 1.要求 完善every函数,如果集合(collection)中的所有对象都存在对应的属性(pre),并且属性(pre)对应的值为真.函数返回ture.反之,返回 ...
- Linux问题分析或解决_samba无法连接
1. windows设置方面问题 问题:window能连接部分服务器的samba共享,一部分无法连接.报错如截图. 解决:前提---其他人连接都没有问题,发现有问题的连接服务器的电脑是win10,而w ...