Disclaimer: This article is a blogger original article, reproduced, please indicate the original source. https://blog.csdn.net/woay2008/article/details/84490842
In the "UNIX Network Programming Volume 1," In this section , we have used the C language using epoll of a TCP echo server program. Now we use Python to achieve the same function, the code is as follows:
# -*- coding: utf-8 -*-
import os
import socket
import select
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
s.setblocking(False)
s.bind(('127.0.0.1', 22222))
s.listen(5)
sockets = {}
epoll = select.epoll()
epoll.register(s.fileno(), select.EPOLLIN)
while True:
events = epoll.poll()
for fd, event in events:
if fd == s.fileno(): # 新的连接请求
c, addr = s.accept()
c.setblocking(False) # 非阻塞
sockets[c.fileno()] = c
epoll.register(c.fileno(), select.EPOLLIN) # 水平触发的方式
else: # 新的数据
c = sockets[fd]
close = False
try:
data = c.recv(1024)
except socket.error:
close = True
if data:
try:
c.send(data)
except socket.error:
close = True
else:
close = True
if close: # socket上出现错误或对端正常关闭连接时,关闭socket
epoll.unregister(fd)
del sockets[fd]
c.close()
Note that the above is implemented using epoll level trigger mode, if you are using epoll edge trigger mode, then the code follows:
# -*- coding: utf-8 -*-
import os
import errno
import socket
import select
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
s.setblocking(False)
s.bind(('127.0.0.1', 22222))
s.listen(5)
sockets = {}
epoll = select.epoll()
epoll.register(s.fileno(), select.EPOLLIN)
while True:
events = epoll.poll()
for fd, event in events:
if fd == s.fileno(): # 新的连接请求
c, addr = s.accept()
c.setblocking(False) # 非阻塞
sockets[c.fileno()] = c
epoll.register(c.fileno(), select.EPOLLIN | select.EPOLLET) # 边沿触发的方式
else: # 新的数据
c = sockets[fd]
close = False
while True:
try:
data = c.recv(1024)
except socket.error, err:
if err[0] != errno.EAGAIN:
close = True
break
if data:
try:
c.send(data)
except socket.error, err:
if err[0] != errno.EAGAIN:
close = True
break
else:
close = True
break
if close: # socket上出现错误或对端正常关闭连接时,关闭socket
epoll.unregister(fd)
del sockets[fd]
c.close()
The code above is just a reference, there are some details not considered. For example, send
when sending data, if the return EAGAIN
, under the right circumstances how to deal with?