尝试在TCP

时间:2018-02-25 02:00:59

标签: c++ linux sockets network-programming tcp-ip

我正在编写一个客户端/服务器应用程序,客户端和服务器应该通过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()函数自己尝试多次?或者在我再试一次之前,我是否需要对套接字进行重置以重置所有内容?我希望这不是一个愚蠢的问题,我找不到任何有关如何多次连接的信息。

提前致谢!

2 个答案:

答案 0 :(得分:1)

  

经过一段时间后,我可以强制connect()返回吗?

没有。您必须将套接字置于非阻塞模式,然后在等待套接字连接时使用select()(e)poll()来提供超时逻辑。如果连接失败或连接时间过长,请关闭套接字,创建一个新套接字,然后重试。

  

或者有没有办法让connect()函数尝试多次?

没有。每次呼叫只能执行1次连接尝试。

  

或者,在我再试一次之前,我是否需要对套接字进行重置以重置所有内容?

无法保证您甚至可以在同一个套接字上多次呼叫connect()。在某些平台上,您必须先销毁套接字并创建新套接字,然后再次调用connect()。你应养成为所有平台做到这一点的习惯。

答案 1 :(得分:0)

将套接字置于非阻塞模式,并使用select()来实现超时。选择套接字上的可写性。请注意,您可以通过这种方式减少平台连接超时,但不能增加它。

sleep()毫无意义,只是浪费时间。