非阻塞模式下,虽然connect出错,但是getsockopt取得的错误却是0的问题

    调试项目代码时,发现了一个奇怪问题,记录如下:

    非阻塞模式下,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

原文地址:https://www.cnblogs.com/gqtcgq/p/7247026.html