我正在编写一个客户端/服务器应用程序,客户端和服务器应该通过TCP套接字相互发送数据。客户端应该连接到服务器,如果连接失败,它应该等待几秒钟,然后再次尝试连接到它(最多一定次数)。
这是我目前的代码:
const int i_TRIES = 5;
time_t t_timeout = 3000;
int i_port = 5678;
int i_socket;
string s_IP = "127.0.0.1";
for(int i = 0; i < i_TRIES; i++)
{
if((i_socket = socket(AF_INET, SOCK_STREAM, 0)) < 0)
{
cout << "[Client]: Socket creation failed." << endl;
exit(EXIT_FAILURE);
}
memset(&server_address, '0', sizeof(server_address));
server_address.sin_family = AF_INET;
server_address.sin_port = htons(i_port);
if(inet_pton(AF_INET, s_IP.c_str(), &server_address.sin_addr) <= 0)
{
cout << "[Client]: Invalid IP address." << endl;
exit(EXIT_FAILURE);
}
if(connect(i_socket, (struct sockaddr *)&server_address, sizeof(server_address)) < 0)
{
if(i < i_TRIES - 2)
{
cout << "[Client]: Connection to server failed. Trying again in " << t_timeout << " ms." << endl;
close(i_socket);
sleep(t_timeout);
}
else
{
cout << "[Client]: Could not connect to server, exiting." << endl;
exit(EXIT_FAILURE);
}
}
else
{
cout << "[Client]: Successfully connected to server." << endl;
break;
}
}
// do stuff with socket
我遇到的问题是第一次调用connect()按预期工作,如果没有服务器就会失败,然后循环重复,但是,第二次connect()会永久阻塞(或至少很多)比我想要的更长)。最初,我的循环只是在connect()if块(下面的代码)周围,这也导致了同样的问题。之后我在循环中包含了整个套接字设置(上面的代码),但这也没有帮助。我也尝试在连接失败后关闭套接字,但这也无济于事。
循环的首字母:
// other stuff from above here
for(int i = 0; i < i_TRIES; i++)
{
if(connect(i_socket, (struct sockaddr *)&server_address, sizeof(server_address)) < 0)
{
if(i < i_TRIES - 2)
{
cout << "[Client]: Connection to server failed. Trying again in " << t_timeout << " ms." << endl;
sleep(t_timeout);
}
else
{
cout << "[Client]: Could not connect to server, exiting." << endl;
exit(EXIT_FAILURE);
}
}
else
{
cout << "[Client]: Successfully connected to server." << endl;
break;
}
}
// do stuff with socket
经过一段时间后,我可以强制connect()返回吗?或者有没有办法让connect()函数自己尝试多次?或者在我再试一次之前,我是否需要对套接字进行重置以重置所有内容?我希望这不是一个愚蠢的问题,我找不到任何有关如何多次连接的信息。
提前致谢!
答案 0 :(得分:1)
经过一段时间后,我可以强制
connect()
返回吗?
没有。您必须将套接字置于非阻塞模式,然后在等待套接字连接时使用select()
或(e)poll()
来提供超时逻辑。如果连接失败或连接时间过长,请关闭套接字,创建一个新套接字,然后重试。
或者有没有办法让
connect()
函数尝试多次?
没有。每次呼叫只能执行1次连接尝试。
或者,在我再试一次之前,我是否需要对套接字进行重置以重置所有内容?
无法保证您甚至可以在同一个套接字上多次呼叫connect()
。在某些平台上,您必须先销毁套接字并创建新套接字,然后再次调用connect()
。你应养成为所有平台做到这一点的习惯。
答案 1 :(得分:0)
将套接字置于非阻塞模式,并使用select()
来实现超时。选择套接字上的可写性。请注意,您可以通过这种方式减少平台连接超时,但不能增加它。
sleep()
毫无意义,只是浪费时间。