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;
}