以下是一个简单的 socket 本地回环测试程序,包含客户端非阻塞的用 select 重连服务器:

服务器端代码:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>

#define PORT 8888
#define BUFFER_SIZE 1024

int main()
{
    int sockfd, newsockfd, n;
    socklen_t clilen;
    char buffer[BUFFER_SIZE];
    struct sockaddr_in serv_addr, cli_addr;

    sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if (sockfd < 0)
    {
        perror("ERROR opening socket");
        exit(1);
    }

    bzero((char *) &serv_addr, sizeof(serv_addr));
    serv_addr.sin_family = AF_INET;
    serv_addr.sin_addr.s_addr = INADDR_ANY;
    serv_addr.sin_port = htons(PORT);

    if (bind(sockfd, (struct sockaddr *) &serv_addr, sizeof(serv_addr)) < 0)
    {
        perror("ERROR on binding");
        exit(1);
    }

    listen(sockfd, 5);
    clilen = sizeof(cli_addr);

    while (1)
    {
        newsockfd = accept(sockfd, (struct sockaddr *) &cli_addr, &clilen);
        if (newsockfd < 0)
        {
            perror("ERROR on accept");
            exit(1);
        }

        bzero(buffer, BUFFER_SIZE);
        n = read(newsockfd, buffer, BUFFER_SIZE-1);
        if (n < 0)
        {
            perror("ERROR reading from socket");
            exit(1);
        }

        printf("Received message: %s\n", buffer);

        n = write(newsockfd, buffer, strlen(buffer));
        if (n < 0)
        {
            perror("ERROR writing to socket");
            exit(1);
        }

        close(newsockfd);
    }

    close(sockfd);
    return 0;
}

客户端代码:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <sys/select.h>

#define PORT 8888
#define BUFFER_SIZE 1024

int main()
{
    int sockfd, n;
    struct sockaddr_in serv_addr;
    char buffer[BUFFER_SIZE];
    fd_set read_fds;
    struct timeval timeout;
    int fd_flags;

    sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if (sockfd < 0)
    {
        perror("ERROR opening socket");
        exit(1);
    }

    bzero((char *) &serv_addr, sizeof(serv_addr));
    serv_addr.sin_family = AF_INET;
    serv_addr.sin_addr.s_addr = inet_addr("127.0.0.1");
    serv_addr.sin_port = htons(PORT);

    fd_flags = fcntl(sockfd, F_GETFL, 0);
    fcntl(sockfd, F_SETFL, fd_flags | O_NONBLOCK);

    if (connect(sockfd, (struct sockaddr *)&serv_addr, sizeof(serv_addr)) < 0)
    {
        if (errno != EINPROGRESS)
        {
            perror("ERROR connecting");
            exit(1);
        }
    }

    FD_ZERO(&read_fds);
    FD_SET(sockfd, &read_fds);

    timeout.tv_sec = 5;
    timeout.tv_usec = 0;

    n = select(sockfd+1, NULL, &read_fds, NULL, &timeout);
    if (n < 0)
    {
        perror("ERROR in select");
        exit(1);
    }
    else if (n == 0)
    {
        printf("ERROR: Connection timeout\n");
        exit(1);
    }

    if (!FD_ISSET(sockfd, &read_fds))
    {
        printf("ERROR: Socket not ready for reading\n");
        exit(1);
    }

    bzero(buffer, BUFFER_SIZE);
    n = read(sockfd, buffer, BUFFER_SIZE-1);
    if (n < 0)
    {
        perror("ERROR reading from socket");
        exit(1);
    }

    printf("Received message: %s\n", buffer);

    close(sockfd);

    return 0;
}

在客户端代码中,我们使用了非阻塞连接方式,通过 select 函数来检查连接是否成功建立,并设置了超时时间。如果超时或连接失败,则程序会退出。

Linux C Socket 本地回环测试程序:客户端非阻塞重连服务器

原文地址: http://www.cveoy.top/t/topic/nIKU 著作权归作者所有。请勿转载和采集!

免费AI点我,无需注册和登录