调试项目代码时,发现了一个奇怪问题,记录如下:
非阻塞模式下,connect发起建链,返回-1(这在非阻塞模式下是很正常的现象)。然后将该socket的写事件进行监听,在写事件触发后,getsockopt函数获取错误(SO_ERROR)时,没有检测到发生错误(第四个参数返回0),最后在write操作时,发生错误。
原因:因配置文件的问题,导致得到的对端IP地址为空字符串"":
struct sockaddr_in saddr; saddr.sin_addr.s_addr = inet_addr(""); saddr.sin_family = AF_INET; /* "default" family */ saddr.sin_port = htons(22);
但是,在connect返回-1时,没有检测错误码是否为EINPROGRESS:
if(connect(m_hSocket, pAddr, nAddrLen) == 0) { return true; } else { return false; }之后直接开始监听写事件。因为connect出错时,该socket套接字上的写连接已经关闭,所以写事件立即触发。
调用getsockopt时,获取的错误码为0,认为没有错误。
再接下来的write操作时,write写入一个已经关闭的连接,导致触发SIGPIPE信号。
总结:
虽然非阻塞模式下的connect,一般情况下都是返回-1,但是却忘了检测errno是否为EINPROGRESS,就像这个问题,因为地址写错了,connect返回-1,但是此时的错误却是"Network is unreachable",这种情况下就不能再监听该socket上的写事件,并在回调函数中调用getsockopt了,因为getsockopt得不到错误码,只会返回0。