基於TCP的回聲服務端/客戶端:實現文件讀寫

服務端實現邏輯:

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的大小,來減少讀寫的次數。

謝謝閱讀

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