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網絡編程》服務器