accept 文件描述符用尽处理

 if (events[i].data.fd == listenfd)  
            {  
                peerlen = sizeof(peeraddr);  
                connfd = ::accept4(listenfd, (struct sockaddr*)&peeraddr,  
                        &peerlen, SOCK_NONBLOCK | SOCK_CLOEXEC);  

                if (connfd == -1)//函数发生错误  
                {  
                    if (errno == EMFILE)//表示每个进程允许打开的文件描述符数量最大值已经到达  
                    {//遇到这种情况的解决方法是  
                    /* 
                    准备一个空闲的文件描述符。遇到这种情况,先关闭这个空闲文件, 
                    获得一个文件描述符名额;再accept(2)拿到socket连接的文件描述符;随后立刻close(2), 
                    这样就优雅地断开了与客户端的连接; 
                    最后重新打开空闲文件,把“坑”填上,以备再次出现这种情况时使用 
                    */  

                        close(idlefd);  
                        idlefd = accept(listenfd, NULL, NULL);  
                        close(idlefd);  
                        idlefd = open("/dev/null", O_RDONLY | O_CLOEXEC);  
                        continue;  
                    }  
                    else  
                        ERR_EXIT("accept4");  
                }  

转载于:https://www.cnblogs.com/MaAce/p/7755692.html

### 使用 Socket 发送文件描述符 在 Linux 系统中,可以通过 `sendmsg` 和 `recvmsg` 函数来实现文件描述符的传递。这依赖于辅助数据(ancillary data),它允许进程间通信机制携带额外的信息。 为了发送文件描述符,通常会创建一个 UNIX 域套接字,并利用控制消息(control message)结构体中的 `SCM_RIGHTS` 类型来进行操作。下面是一个简单的例子展示如何设置客户端和服务端之间的文件描述符传输: 服务端代码如下所示: ```python import socket import struct import os def send_fd(sock, fd_to_send): anc_data = [(socket.SOL_SOCKET, socket.SCM_RIGHTS, struct.pack("i", fd_to_send))] msg = b"" buf = bytearray() # 不需要实际的数据缓冲区 sock.sendmsg([msg], anc_data) server_sock = socket.socket(socket.AF_UNIX, socket.SOCK_STREAM) server_address = "/tmp/uds_socket" if os.path.exists(server_address): os.unlink(server_address) server_sock.bind(server_address) server_sock.listen(1) conn, addr = server_sock.accept() file_descriptor = open("/path/to/file").fileno() send_fd(conn, file_descriptor) conn.close() server_sock.close() ``` 客户端接收部分可以这样编写: ```python import socket import array def recv_fd(sock): fds = array.array("i") # 创建用于存储传入fd的数组 msg, ancdata, flags, addr = sock.recvmsg(0, socket.CMSG_LEN(fds.buffer_info()[1] * fds.itemsize)) for cmsg_level, cmsg_type, cmsg_data in ancdata: if cmsg_level == socket.SOL_SOCKET and cmsg_type == socket.SCM_RIGHTS: fds.frombytes(cmsg_data[:len(cmsg_data) - (len(cmsg_data) % fds.itemsize)]) if not fds: return None return list(fds)[0] client_sock = socket.socket(socket.AF_UNIX, socket.SOCK_STREAM) client_sock.connect("/tmp/uds_socket") received_fd = recv_fd(client_sock) print(f"Received FD: {received_fd}") with os.fdopen(received_fd, 'r') as f: content = f.read() print(content) client_sock.close() ``` 上述程序展示了怎样通过 Unix Domain Sockets 来交换文件描述符[^2]。注意这里使用的是 AF_UNIX 地址族而不是 AF_INET,因为后者并不支持这种类型的附加信息传送方式。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值