C++回聲服務器_2-修復客戶端問題

C++回聲服務器_1-簡單版本中的問題出在客戶端。客戶端經過write函數一次性發送數據,過一段時間再調用一次read函數,指望接收傳輸的數據。問題在於這段時間究竟是多久?理想的客戶端應在接收到數據時當即讀取數據。c++

改進

客戶端發送數據時,能夠知道數據的大小(長度)。客戶端接收數據的時候,能夠知道接收到數據的大小(長度)。因此,客戶端循環調用read函數,直到接收到的數據總大小(長度)等於發送的數據的大小(長度)時,已完成全部數據的接收。git

修改的代碼:github

// 發送的字符串長度、接收字符串的長度、每次read函數接受到字符串的長度
ssize_t str_len, recv_len, recv_cnt;

str_len = write(sock, message, strlen(message)); // 向服務器發送數據
recv_len = 0;
// 循環調用read函數,直到接收到全部數據爲止
while (recv_len < str_len) {
    recv_cnt = read(sock, message, BUF_SIZE); // 讀取來自客戶端的服務器
    if (recv_cnt == -1) {
        error_handling("read() error");
    }
    recv_len += recv_cnt;
}
message[recv_len] = 0;

完整的客戶端代碼:編程

#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/socket.h>

const int BUF_SIZE = 1024;

void error_handling(const char *message);

// 接收兩個參數,argv[1]爲IP地址,argv[2]爲端口號
int main(int argc, char *argv[]) {
    int sock;
    struct sockaddr_in server_addr;
    char message[BUF_SIZE];
    // 發送的字符串長度、接收字符串的長度、每次read函數接受到字符串的長度
    ssize_t str_len, recv_len, recv_cnt;

    if (argc != 3) {
        printf("Usage : %s <IP> <port>\n", argv[0]);
        exit(1);
    }

    sock = socket(PF_INET, SOCK_STREAM, 0);
    if (sock == -1) {
        error_handling("socket() error");
    }

    // 地址信息初始化
    memset(&server_addr, 0, sizeof(server_addr));
    server_addr.sin_family = AF_INET; // IPV4 地址族
    server_addr.sin_addr.s_addr = inet_addr(argv[1]); // 服務器IP地址
    server_addr.sin_port = htons(atoi(argv[2])); // 服務器端口號

    // 向服務器發送鏈接請求
    if (connect(sock, (struct sockaddr*)&server_addr, sizeof(server_addr)) == -1) {
        error_handling("connect() error");
    } else {
        printf("Connect...");
    }

    while (1) {
        printf("Input message( Q to quit ): ");
        fgets(message, BUF_SIZE, stdin);

        // 若是輸入q或者Q,則退出
        if (!strcmp(message, "q\n") || !strcmp(message, "Q\n")) {
            break;
        }

        str_len = write(sock, message, strlen(message)); // 向服務器發送數據
        recv_len = 0;
        // 循環調用read函數,直到接收到全部數據爲止
        while (recv_len < str_len) {
            recv_cnt = read(sock, message, BUF_SIZE); // 讀取來自客戶端的服務器
            if (recv_cnt == -1) {
                error_handling("read() error");
            }
            recv_len += recv_cnt;
        }
        message[recv_len] = 0;

        printf("Message from server: %s \n", message);
    }
    // 關閉鏈接
    close(sock);

    return 0;
}

項目代碼

githubsegmentfault

參考

《TCP/IP網絡編程》服務器

相關文章
相關標籤/搜索