Linux 网络编程 socket 错误分析
socket 错误码:
EINTR: 4
阻塞的操作被取消阻塞的调用打断。如设置了发送接收超时,就会遇到这种错误。
只能针对阻塞模式的 socket。读,写阻塞的 socket 时,-1 返回,错误号为 INTR。另外,如
果出现 EINTR 即 errno 为 4,错误描述 Interrupted system call,操作也应该继续。如果 recv
的返回值为 0,那表明连接已经断开,接收操作也应该结束。
ETIMEOUT:110
1、操作超时。一般设置了发送接收超时,遇到网络繁忙的情况,就会遇到这种错误。
2、服务器做了读数据做了超时限制,读时发生了超时。
3、错误被描述为“connect time out”,即“连接超时”,这种情况一般发生在服务器主机崩
溃。此时客户 TCP 将在一定时间内(依具体实现)持续重发数据分节,试图从服务 TCP 获
得一个 ACK 分节。当最终放弃尝试后(此时服务器未重新启动),内核将会向客户进程返
回 ETIMEDOUT 错误。如果某个中间路由器判定该服务器主机已经不可达,则一般会响应
“destination unreachable”-“目的地不可达”的 ICMP 消息,相应的客户进程返回的错误
是 EHOSTUNREACH 或 ENETUNREACH。当服务器重新启动后,由于 TCP 状态丢失,
之前所有的连接信息也不存在了,此时对于客户端发来请求将回应 RST。如果客户进程对
检测服务器主机是否崩溃很有必要,要求即使客户进程不主动发送数据也能检测出来,那么
需要使用其它技术,如配置 SO_KEEPALIVE Socket 选项,或实现某些心跳函数。
EAGAIN:
1、Send 返回值小于要发送的数据数目,会返回 EAGAIN 和 EINTR。
2、recv 返回值小于请求的长度时说明缓冲区已经没有可读数据,但再读不一定会触发
EAGAIN,有可能返回 0 表示 TCP 连接已被关闭。
3、当 socket 是非阻塞时,如返回此错误,表示写缓冲队列已满,可以做延时后再重试.
4、在 Linux 进行非阻塞的 socket 接收数据时经常出现 Resource temporarily unavailable,errno
代码为 11(EAGAIN),表明在非阻塞模式下调用了阻塞操作,在该操作没有完成就返回这个
错误,这个错误不会破坏 socket 的同步,不用管它,下次循环接着 recv 就可以。对非阻塞
socket 而言,EAGAIN 不是一种错误。
EPIPE:
1、Socket 关闭,但是 socket 号并没有置-1。继续在此 socket 上进行 send 和 recv,就会返回
这种错误。这个错误会引发 SIGPIPE 信号,系统会将产生此 EPIPE 错误的进程杀死。所以,
一般在网络程序中,首先屏蔽此消息,以免发生不及时设置 socket 进程被杀死的情况。
2、write(..) on a socket that has been closed at the other end will cause a SIGPIPE.
3、错误被描述为“broken pipe”,即“管道破裂”,这种情况一般发生在客户进程不理会(或
未及时处理)Socket 错误,继续向服务 TCP 写入更多数据时,内核将向客户进程发送
SIGPIPE 信号,该信号默认会使进程终止(此时该前台进程未进行 core dump)。结合上边
的 ECONNRESET 错误可知,向一个 FIN_WAIT2 状态的服务 TCP(已 ACK 响应 FIN
分节)写入数据不成问题,但是写一个已接收了 RST 的 Socket 则是一个错误。