python使用epoll實現的服務端例子

#!/usr/bin/python
# -*- coding: UTF-8 -*-
import socket
import select

send_data = "hello world!"
send_len = len(send_data)
recv_len = 1024
tcp_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
tcp_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
addr = ("0.0.0.0", 8765)
tcp_socket.bind(addr)
tcp_socket.listen(5)
tcp_socket.setblocking(False)
epoll = select.epoll()
'''(邊緣觸發)select.EPOLLIN | select.EPOLLET'''
epoll.register(tcp_socket.fileno(), select.EPOLLIN)
'''因爲epoll返回的觸發事件對應的是套接字文件描述符,所以需要在字典中加入對應關係'''
fd_to_socket = {tcp_socket.fileno():tcp_socket}

while True :
    events = epoll.poll(-1)
    for fd, event in events:
        fd_socket = fd_to_socket[fd]
        if fd == tcp_socket.fileno():
            while True:
                try:
                    new_socket, new_addr = fd_socket.accept()
                except socket.error as e:
                    (errno, err_msg) = e
                    print errno
                    print err_msg
                    if errno == 11:
                        break
                print "new accpet:", new_addr
                new_socket.setblocking(False)
                new_socket.setsockopt(socket.IPPROTO_TCP, socket.TCP_NODELAY, 1)
                epoll.register(new_socket.fileno(), select.EPOLLIN)
                fd_to_socket[new_socket.fileno()] = new_socket
        elif event&select.EPOLLIN:
            recv_datas = []
            recd = 0
            while (recd < recv_len):
                try:
                    recv_data = fd_socket.recv(recv_len - recd)
                    '''處理讀的正常關閉'''
                    if recv_data == "":
                        print "close socket"
                        epoll.unregister(fd)
                        fd_to_socket[fd].close()
                        del fd_to_socket[fd]
                        break
                    else:
                        recv_datas.append(recv_data)
                        recd = recd + len(recv_data)
                '''處理異常關閉(EAGAIN,EINTR)'''
                except socket.error as e:
                    (errno, err_msg) = e
                    print errno
                    print err_msg
                    '''因爲用的水平觸發,EAGAIN我們跳出循環,等待下次觸發再讀就好了'''
                    if errno == 11:
                        break
                    '''軟中斷打斷了還要繼續讀'''
                    elif errno == 4:
                        continue
                    '''其它錯誤我們直接關閉套接字'''
                    else:
                        print "close socket"
                        epoll.unregister(fd)
                        fd_to_socket[fd].close()
                        del fd_to_socket[fd]
                        break
                print repr(recv_datas)
                total_send = 0
                while total_send < send_len:
                    sent = fd_socket.send(send_data[total_send:])
                    if sent == 0:
                        print "close socket"
                        epoll.unregister(fd)
                        fd_to_socket[fd].close()
                        del fd_to_socket[fd]
                        break
                    else:
                        print repr(send_data[total_send:])
                        total_send = total_send + sent

其實這裏的異常處理我們也可以用

except IOError as e:
        print e.errno
        print e.strerror
發佈了68 篇原創文章 · 獲贊 74 · 訪問量 17萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章