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

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

改進

客戶端發送數據時,可以知道數據的大小(長度)。客戶端接收數據的時候,可以知道接收到數據的大小(長度)。所以,客戶端循環調用read函數,直到接收到的數據總大小(長度)等於發送的數據的大小(長度)時,已完成所有數據的接收。

修改的代碼:

// 發送的字符串長度、接收字符串的長度、每次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[0]爲IP地址,argv[1]爲端口號
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;
}

項目代碼

github

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章