服務端實現邏輯:
1、接收來自客戶端的連接請求
2、接受來自客戶端的文件名稱信息
3、根據文件名稱打開對應文件,並讀取文件內容
4、將文件內容發送給客戶端
客戶端實現邏輯:
1、建立對服務端的TCP連接
2、接收用戶需要讀取的文件名,併發送給服務端
3、接收來自服務端的文件內容或讀取失敗的回覆信息
服務端代碼實現:echo_server.cpp
#include <iostream>
#include <string.h>
#include <fcntl.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/socket.h>
using namespace std;
#define BUF_SIZE 100
int main(int argc, char* argv[])
{
char message[BUF_SIZE] = { 0 };
// PF_INET -- IPv4協議族 SOCK_STREAM -- 流式控制協議
int serv_sock = socket(PF_INET, SOCK_STREAM, 0);
// 初始化地址信息
struct sockaddr_in serv_addr;
memset(&serv_addr, 0, sizeof(serv_addr));
serv_addr.sin_family = AF_INET; // IPv4地址族
serv_addr.sin_addr.s_addr = htonl(INADDR_ANY);
serv_addr.sin_port = htons(atoi(argv[1]));
// 地址信息綁定
bind(serv_sock, (struct sockaddr*)&serv_addr, sizeof(serv_addr));
// 開始監聽
listen(serv_sock, 5);
// 接收來自客戶端的連接請求
struct sockaddr_in clnt_addr;
socklen_t clnt_adr_sz = sizeof(clnt_addr);
int clnt_sock = accept(serv_sock, (struct sockaddr*)&clnt_addr, &clnt_addr_sz);
if(clnt_sock != -1)
{
cout << "client connect!!!" << endl;
}
// 讀取來自客戶端的信息:即文件名
read(clnt_sock, message, sizeof(message));
cout << "file name = " << message << endl;
// 讀取文件內容
int fd = open(message, O_RDONLY);
if(fd != -1)
{
int read_len = 0;
int write_count = 0;
while((read_len = read(fd,message, sizeof(message) - 1)) != 0)
{
++write_count;
message[read_len] = 0;
write(clnt_sock, message, read_len);
}
cout << "write over! write count: " << write_count << endl;
}
else
{
cout << "file open error!!!" << endl;
}
// 關閉socket
close(clnt_sock);
close(serv_sock);
return 0;
}
客戶端代碼實現:echo_client.cpp
#include <iostream>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/socket.h>
using namespace std;
#define BUF_SIZE 100
int main(int argc, char* argv[])
{
char message[BUF_SIZE] = { 0 };
// 建立socket
int sock = socket(PF_INET, SOCK_STREAM, 0);
// 初始化地址信息
struct sockaddr_in serv_addr;
memset(&serv_addr, 0, sizeof(serv_addr));
serv_addr.sin_family = AF_INET;
serv_addr.sin_addr.s_addr = inet_addr(argv[1]);
serv_addr.sin_port = htons(atoi(argv[2]));
// 建立對服務端的連接
if(connect(sock,(struct sockaddr*)&serv_addr, sizeof(serv_addr)) != -1)
{
cout << "connected..." << endl;
}
// 用戶輸入文件名
string file_name;
cout << "please input file name you want read: " << endl;
cin >> file_name;
cout << "file_name confirm: " << file_name << endl;
// 向服務端發送文件名稱
write(sock, file_name.c_str(), file_name.length());
// 讀取來自服務端的文件內容
int read_len = 0;
int read_count = 0;
while((read_len = read(sock, message, BUF_SIZE - 1)) != 0)
{
message[read_len] = 0;
cout << "=============================" << endl;
cout << "read count: " << ++read_count << endl;
cout << "file content: " << message << endl;
cout << "=============================" << endl;
}
// 關閉sock請求
close(sock);
return 0;
}
運行服務端和客戶端程序,注意輸入參數:
服務端:可執行程序 + 監聽的端口號
客戶端:可執行程序 + 服務端IP + 服務端端口號
以下是將要讀取的文件內容:
以下是服務端運行結果:
以下是客戶端運行結果:
考慮到不同的文件大小不同,所以採用服務端程序循環讀取和發送
,客戶端也採用循環接收,並且設置count變量,來表示服務端讀文件的次數和客戶端讀取來自服務端消息的次數。
另外,可根據文件大小不同,調整BUF_SIZE的大小,來減少讀寫的次數。
謝謝閱讀