python并发原理(阻塞、非阻塞、epoll)
2018-07-17 04:06:02来源:博客园 阅读 ()
在Linux系统中
01 阻塞服务端
特征:1对1,阻塞。
1 import socket 2 3 server = socket.socket() #生成套接字对象 4 server.bind(('0.0.0.0', 8000)) #套接字绑定ip和端口,变为监听套接字 5 server.listen(5) #开始监听 6 7 while True: 8 conn, addr = server.accept() #建立连接,生成对等套接字 9 print('用户连接:', addr) 10 while True: 11 try: 12 data = conn.recv(1024) 13 if data == b'Q' or data == b'q': 14 print('用户退出:', addr) 15 break 16 else: 17 print('收到的消息:', data.decode()) 18 conn.send(data) 19 except Exception as e: 20 print(e) 21 conn.close() 22 break
02 非阻塞服务端
特征:1对多,轮询,非阻塞,占用资源多。
1 import socket 2 3 server = socket.socket() #创建套接字 4 server.setblocking(False) #把套接字设置为非阻塞 5 server.bind(('0.0.0.0', 8001)) #绑定IP和端口 6 server.listen(5) #监听端口 7 8 9 all_connection = [] #保存已经连接的客户 10 while True: 11 #只管连接的事情 12 try: 13 conn, addr = server.accept() # 建立连接,没有就抛出异常 14 conn.setblocking(False) #设置非阻塞 15 print('用户连接:', addr) 16 all_connection.append(conn) 17 except Exception as e: 18 pass 19 20 21 #处理已经连接用户的消息 22 handle = all_connection.copy() #完全拷贝了列表 23 for connection in handle: 24 try: 25 recv_data = connection.recv(1024) 26 if recv_data: 27 print(recv_data.decode()) 28 connection.send(recv_data) 29 else: #客户端消息处理完了,已经断开了连接 30 print('断开连接', connection) 31 connection.close() 32 all_connection.remove(connection) #从客户列表里移除断开连接的客户 33 except Exception as e: 34 pass
03 epoll服务端
特征:1对多,通知机制,非阻塞,占用资源少;
epoll:注册惰性事件回调。
1 import selectors #调用epoll的模块 2 import socket 3 4 epoll = selectors.EpollSelector() #生成一个epoll 5 server = socket.socket() #生成套接字 6 server.bind(('', 8082)) #参数1‘’与‘0.0.0.0’等价,表示ip都可接入 7 server.listen(100) 8 9 10 #回调函数 11 def create_conneciton(server): 12 #百分百有人连接,不会阻塞 13 conn, addres = server.accept() #生成对等连接套接字 14 15 16 #处理消息的函数注册 17 epoll.register(conn, selectors.EVENT_READ, read_data) 18 return conn 19 20 21 22 #回调函数 处理消息 23 def read_data(conn): 24 25 data = conn.recv(1024) 26 if data: 27 print(data) 28 conn.send(data) 29 else: 30 epoll.unregister(conn) #删掉注册事件 31 32 33 #1 34 #把监听套接字和生成对等套接字的函数注册到read事件(有用户连接) 35 epoll.register(server, selectors.EVENT_READ, create_conneciton) 36 37 38 #2 39 while True: #事件循环 40 events = epoll.select() #去操作系统查询 41 42 for key,mask in events: 43 sock = key.fileobj #连接客户端的套接字 44 callback = key.data #回调函数 45 46 #read_data(conn), create_conneciton(server) 47 callback(sock) #调用函数
04 客户端
测试服务端。
1 import socket 2 3 client = socket.socket() 4 client.connect(('127.0.0.1', 8082)) 5 6 while True: 7 data = input('输入数据:') 8 client.send(data.encode()) 9 recv_data = client.recv(1024) 10 print(recv_data.decode())
标签:
版权申明:本站文章部分自网络,如有侵权,请联系:west999com@outlook.com
特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系,版权归原作者所有
上一篇:原生sql整理
下一篇:python?自动发送邮件
- python3基础之“术语表(2)” 2019-08-13
- python3 之 字符串编码小结(Unicode、utf-8、gbk、gb2312等 2019-08-13
- Python3安装impala 2019-08-13
- 小白如何入门 Python 爬虫? 2019-08-13
- python_字符串方法 2019-08-13
IDC资讯: 主机资讯 注册资讯 托管资讯 vps资讯 网站建设
网站运营: 建站经验 策划盈利 搜索优化 网站推广 免费资源
网络编程: Asp.Net编程 Asp编程 Php编程 Xml编程 Access Mssql Mysql 其它
服务器技术: Web服务器 Ftp服务器 Mail服务器 Dns服务器 安全防护
软件技巧: 其它软件 Word Excel Powerpoint Ghost Vista QQ空间 QQ FlashGet 迅雷
网页制作: FrontPages Dreamweaver Javascript css photoshop fireworks Flash