如何确保TCP套接字在创建时连接

时间:2011-07-22 03:25:15

标签: c++ sockets winsock tcpsocket

我有一个发送TCP数据包的客户端应用程序。目前,我的应用程序这样做: 创建一个套接字,绑定并发送数据包。

struct sockaddr_in localaddress;
localaddress.sin_port = htons(0);
localaddress.sin_addr.s_addr = INADDR_ANY;

int socket;
socket = socket(AF_INET, SOCK_STREAM, 0));
bind(socket, (struct sockaddr *)&sin,sizeof(struct sockaddr_in) );

在另一个线程中,应用程序连接并发送数据包:

struct socketaddr_in remoteaddress;
// omitted: code to set the remote address/ port etc...
nRet = connect (socket, (struct sockaddr * ) & remoteaddress, sizeof (sockaddr_in ));
if (nRet == -1)
    nRet = WSAGetLastError();
if (nRet == WSAEWOULDBLOCK) {
    int errorCode = 0;
    socklen_t codeLen = sizeof(int);
    int retVal = getsockopt(
            socket, SOL_SOCKET, SO_ERROR, ( char * ) &errorCode, &codeLen );
    if (errorCode == 0 && retVal != 0)
        errorCode = errno;
}
/* if the connect succeeds, program calls a callback function to notify the socket is connected, which then calls send() */

现在我想指定本地端口的端口范围,所以我将代码更改为

nPortNumber = nPortLow;
localaddress.sin_port = htons(nPortNumber);

并在我的端口范围内循环nPortNumber,例如(4000 - 5000),直到绑定成功。

由于我总是从低端口启动我的nPortNumber,如果先前在同一端口上创建了一个套接字,我将WSAEADDRINUSE错误视为errorCode,这对我来说太迟了,因为它已经通过了套接字创建阶段。 (为什么我没有在bind()或connect()上获得WSAEADDRINUSE?)

有没有办法可以提前获得WSAEADDRINUSE,或者有没有办法在绑定和连接的端口范围内创建套接字?

提前致谢!

1 个答案:

答案 0 :(得分:0)

我无法100%肯定地回答,我应该知道你实际得到WSAEADDRINUSE

在任何情况下,您都不能在绑定时获取它,因为您使用INADDR_ANY。 IIRC,这实际上将绑定过程延迟到实际连接(我的猜测是它然后根据远程地址的路由更改INADDR)。但是,据我所知,您应该在连接调用时实际收到错误...