C++实现FTP综合应用详解

 更新时间:2022年08月02日 17:22:05   作者:十梦九有妳  
这篇文章主要为大家详细介绍了C++实现FTP综合应用,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

本文为大家分享了FTP综合应用编程(C++),供大家参考,具体内容如下

1.学校实验

借鉴了网上一位大佬的作品,然后自己改改拿来完成了算是还行的作品。

代码一共大概是900多行,可以直接粘贴下来看看,其实还是很容易理解的。

运行的截图附上:

客户端:

服务器端:

2.服务器端代码

头文件:(sizes.h)

#pragma once
 
//服务器侦听控制连接请求的端口
#define CMD_PORT 5858
//客户机侦听数据连接请求的端口
#define DATA_PORT 5850
//命令报文参数缓存的大小
#define CMD_PARAM_SIZE 256
//回复报文消息缓存的大小
#define RSPNS_TEXT_SIZE 256
#define BACKLOG 10
#define DATA_BUFSIZE 4096
 
//命令类型
typedef enum {
    LS, PWD, CD, DOWN, UP, QUIT
} CmdID;
 
//命令报文,从客户端发往服务器
typedef struct _CmdPacket {
    CmdID cmdid;
    char param[CMD_PARAM_SIZE];
} CmdPacket;
 
//回复报文的类型
typedef enum {
    OK, ERR
} RspnsID;
 
//回复报文,从服务器发往客户端
typedef struct _RspnsPacket {
    RspnsID rspnsid;
    char text[RSPNS_TEXT_SIZE];
} RspnsPacket;

源文件:(服务器端.cpp)

#include <WinSock2.h>
#include "sizes.h"
#include <iostream>
 
#pragma comment(lib, "ws2_32.lib")
 
//创建线程时传递的数据结构,内含控制连接套接字和客户端地址信息:
struct threadData {
    SOCKET tcps;
    sockaddr_in clientaddr;
};
 
//全局函数声明:
//FTP初始化,创建一个侦听套接字:
int InitFTP(SOCKET *pListenSock);
int InitDataSocket(SOCKET *pDatatcps, SOCKADDR_IN *pClientAddr);
int ProcessCmd(SOCKET tcps, CmdPacket* pCmd, SOCKADDR_IN *pClientAddr);
int SendRspns(SOCKET tcps, RspnsPacket* prspns);
int RecvCmd(SOCKET tcps, char* pCmd);
int SendFileList(SOCKET datatcps);
int SendFileRecord(SOCKET datatcps, WIN32_FIND_DATA* pfd);
int SendFile(SOCKET datatcps, FILE* file);
int RecvFile(SOCKET datatcps, char* filename);
int FileExists(const char *filename);
 
//线程函数,参数包括相应控制连接的套接字:
DWORD WINAPI ThreadFunc(LPVOID lpParam) {
    SOCKET tcps;
    sockaddr_in clientaddr;
    tcps = ((struct threadData *)lpParam)->tcps;
    clientaddr = ((struct threadData *)lpParam)->clientaddr;
    printf("socket的编号是:%u.\n", tcps);
 
    //发送回复报文给客户端,内含命令使用说明:
    printf("Serve client %s:%d\n", inet_ntoa(clientaddr.sin_addr), ntohs(clientaddr.sin_port));
    RspnsPacket rspns = { OK,
            "欢迎进入FTP综合应用系统!\n"
            "你可以使用的命令:\n"
            "ls\t<展示当前目录下的文件(夹),无需参数>\n"
            "pwd\t<展示当前目录的绝对路径,无需参数>\n"
            "cd\t<切换到指定目录,参数为路径>\n"
            "down\t<下载文件,参数为文件名>\n"
            "up\t<上传文件,参数为文件名>\n"
            "quit\t<退出系统,无需参数>\n"
    };
    SendRspns(tcps, &rspns);
 
    //循环获取客户端命令报文并进行处理
    for (;;) {
        CmdPacket cmd;
        if (!RecvCmd(tcps, (char *)&cmd))
            break;
        if (!ProcessCmd(tcps, &cmd, &clientaddr))
            break;
    }
 
    //线程结束前关闭控制连接套接字:
    closesocket(tcps);
    delete lpParam;
    return 0;
}
 
int main(int argc, char* argv[]) {
    SOCKET tcps_listen;  //FTP服务器控制连接侦听套接字
    struct threadData *pThInfo;
 
    if (!InitFTP(&tcps_listen))  //FTP初始化
        return 0;
    printf("FTP服务器开始监听,端口号为:%d。。。。。。\n", CMD_PORT);
 
    //循环接受客户端连接请求,并生成线程去处理:
    for (;;) {
        pThInfo = NULL;
        pThInfo = new threadData;
        if (pThInfo == NULL) {
            printf("为新线程申请空间失败。\n");
            continue;
        }
 
        int len = sizeof(struct threadData);
        //等待接受客户端控制连接请求
        pThInfo->tcps = accept(tcps_listen, (SOCKADDR*)&pThInfo->clientaddr, &len);
 
        //创建一个线程来处理相应客户端的请求:
        DWORD dwThreadId, dwThrdParam = 1;
        HANDLE hThread;
 
        hThread = CreateThread(
            NULL,               //无需安全性的继承
            0,                    //默认线程栈大小
            ThreadFunc,            //线程入口函数
            pThInfo,            //线程入口函数的参数
            0,                    //立即启动线程
            &dwThreadId);        //返回线程的id值
 
        //检查返回值是否创建线程成功
        if (hThread == NULL) {
            printf("创建线程失败。\n");
            closesocket(pThInfo->tcps);
            delete pThInfo;
        }
    }
 
    return 0;
 
}
 
//FTP初始化,创建一个侦听套接字:
 
int InitFTP(SOCKET *pListenSock) {
    //按照此步骤创建新的服务器端套接字,嗯,没错,前三个都是这个步骤
    //startup->socket->bind->listen
    WORD wVersionRequested;
    WSADATA wsaData;
    int err;
    SOCKET tcps_listen;
 
    wVersionRequested = MAKEWORD(2, 2);
    err = WSAStartup(wVersionRequested, &wsaData);
    if (err != 0) {
        printf("Winsock初始化时发生错误!\n");
        return 0;
    }
    if (LOBYTE(wsaData.wVersion) != 2 || HIBYTE(wsaData.wVersion) != 2) {
        WSACleanup();
        printf("无效Winsock版本!\n");
        return 0;
    }
 
    tcps_listen = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
    if (tcps_listen == INVALID_SOCKET) {
        WSACleanup();
        printf("创建Socket失败!\n");
        return 0;
    }
 
    SOCKADDR_IN tcpaddr;
    tcpaddr.sin_family = AF_INET;
    tcpaddr.sin_port = htons(CMD_PORT);
    tcpaddr.sin_addr.S_un.S_addr = htonl(INADDR_ANY);
    err = bind(tcps_listen, (SOCKADDR*)&tcpaddr, sizeof(tcpaddr));
    if (err != 0) {
        err = WSAGetLastError();
        WSACleanup();
        printf("Scoket绑定时发生错误!\n");
        return 0;
    }
    err = listen(tcps_listen, 3);
    if (err != 0) {
        WSACleanup();
        printf("Scoket监听时发生错误!\n");
        return 0;
    }
 
    *pListenSock = tcps_listen;
    return 1;
}
 
 
//建立数据连接
//pDatatcps:用于存储数据连接套接字
//pClientAddr:指向客户端的控制连接套接字地址,需要使用其中的IP地址
//返回值:0表示失败,1正常
int InitDataSocket(SOCKET *pDatatcps, SOCKADDR_IN *pClientAddr) {
    SOCKET datatcps;
 
    //创建socket
    datatcps = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
    if (datatcps == INVALID_SOCKET) {
        printf("Creating data socket failed!\n");
        return 0;
    }
 
    SOCKADDR_IN tcpaddr;
    memcpy(&tcpaddr, pClientAddr, sizeof(SOCKADDR_IN));
    tcpaddr.sin_port = htons(DATA_PORT);    //如若有什么意外只需要在头文件修改端口值
 
    //请求连接客户端
    if (connect(datatcps, (SOCKADDR*)&tcpaddr, sizeof(tcpaddr)) == SOCKET_ERROR) {
        printf("Connecting to client failed!\n");
        closesocket(datatcps);
        return 0;
    }
 
    *pDatatcps = datatcps;
    return 1;
}
 
//处理命令报文
//tcps:控制连接套接字
//pcmd:指向待处理的命令报文
//pClientAddr:指向客户端控制连接套接字地址
//返回值:0表示有错或者需要结束连接,1正常
 
int ProcessCmd(SOCKET tcps, CmdPacket* pCmd, SOCKADDR_IN *pClientAddr) {
    SOCKET datatcps;   //数据连接套接字
    RspnsPacket rspns;  //回复报文
    FILE* file;
 
    //根据命令类型分派执行:
    switch (pCmd->cmdid) {
    case LS://展示当前目录下的文件列表
        //首先建立数据连接:
        if (!InitDataSocket(&datatcps, pClientAddr))
            return 0;
        //发送文件列表信息:
        if (!SendFileList(datatcps))
            return 0;
        break;
    case PWD://展示当前目录的绝对路径
        rspns.rspnsid = OK;
        //获取当前目录,并放至回复报文中
        if (!GetCurrentDirectory(RSPNS_TEXT_SIZE, rspns.text))
            strcpy(rspns.text, "Can't get current dir!\n");
        if (!SendRspns(tcps, &rspns))
            return 0;
        break;
    case CD://设置当前目录,使用win32 API 接口函数
        if (SetCurrentDirectory(pCmd->param)) {
            rspns.rspnsid = OK;
            if (!GetCurrentDirectory(RSPNS_TEXT_SIZE, rspns.text))
                strcpy(rspns.text, "切换当前目录成功!但是不能获取到当前的文件列表!\n");
        }
        else {
            strcpy(rspns.text, "不能更换到所选目录!\n");
        }
        if (!SendRspns(tcps, &rspns))   //发送回复报文
            return 0;
        break;
    case DOWN://处理下载文件请求:
        file = fopen(pCmd->param, "rb");   //打开要下载的文件
        if (file) {
            rspns.rspnsid = OK;
            sprintf(rspns.text, "下载文件%s\n", pCmd->param);
            if (!SendRspns(tcps, &rspns)) {
                fclose(file);
                return 0;
            }
            else {
                //创建额外的数据连接来传送数据:
                if (!InitDataSocket(&datatcps, pClientAddr)) {
                    fclose(file);
                    return 0;
                }
                if (!SendFile(datatcps, file))
                    return 0;
                fclose(file);
            }
        }
        else  //打开文件失败
        {
            rspns.rspnsid = ERR;
            strcpy(rspns.text, "不能打开文件!\n");
            if (!SendRspns(tcps, &rspns))
                return 0;
        }
        break;
    case UP://处理上传文件请求
        //首先发送回复报文
        char filename[64];
        strcpy(filename, pCmd->param);
        //首先看一下服务器上是否已经有这个文件里,如果有就告诉客户端不用传输了
        if (FileExists(filename)) {
            rspns.rspnsid = ERR;
            sprintf(rspns.text, "服务器已经存在名字为%s的文件!\n", filename);
            if (!SendRspns(tcps, &rspns))
                return 0;
        }
        else {
            rspns.rspnsid = OK;
            if (!SendRspns(tcps, &rspns))
                return 0;
            //另建立一个数据连接来接受数据:
            if (!InitDataSocket(&datatcps, pClientAddr))
                return 0;
            if (!RecvFile(datatcps, filename))
                return 0;
        }
        break;
    case QUIT:
        printf("客户端断开连接。\n");
        rspns.rspnsid = OK;
        strcpy(rspns.text, "常来啊!\n");
        SendRspns(tcps, &rspns);
        return 0;
 
 
    }
 
    return 1;
 
}
 
//发送回复报文
int SendRspns(SOCKET tcps, RspnsPacket* prspns) {
    if (send(tcps, (char *)prspns, sizeof(RspnsPacket), 0) == SOCKET_ERROR) {
        printf("与客户端失去连接。\n");
        return 0;
    }
    return 1;
}
 
//接收命令报文
//tcps:控制连接套接字
//pCmd:用于存储返回的命令报文
//返回值:0表示有错或者连接已经断开,1表示正常
int RecvCmd(SOCKET tcps, char* pCmd) {                    //used to receive command from client
    int nRet;
    int left = sizeof(CmdPacket);
 
    //从控制连接中读取数据,大小为 sizeof(CmdPacket):
    while (left) {
        nRet = recv(tcps, pCmd, left, 0);
        if (nRet == SOCKET_ERROR) {
            printf("从客户端接受命令时发生未知错误!\n");
            return 0;
        }
        if (!nRet) {
            printf("客户端关闭了连接!\n");
            return 0;
        }
 
        left -= nRet;
        pCmd += nRet;
    }
    return 1;   //成功获取命令报文
}
 
 
//发送一项文件信息:
int SendFileRecord(SOCKET datatcps, WIN32_FIND_DATA* pfd) {                    //used to send response to client
    char filerecord[MAX_PATH + 32];
    FILETIME ft;
    FileTimeToLocalFileTime(&pfd->ftLastWriteTime, &ft);
    SYSTEMTIME lastwtime;
    FileTimeToSystemTime(&ft, &lastwtime);
    char* dir = (char*)(pfd->dwFileAttributes&FILE_ATTRIBUTE_DIRECTORY ? "<DIR>" : "");
    sprintf(filerecord, "%04d-%02d-%02d%02d:%02d   %5s   %10d   %-20s\n",
        lastwtime.wYear,
        lastwtime.wMonth,
        lastwtime.wDay,
        lastwtime.wHour,
        lastwtime.wMinute,
        dir,
        pfd->nFileSizeLow,
        pfd->cFileName);
    if (send(datatcps, filerecord, strlen(filerecord), 0) == SOCKET_ERROR) {
        printf("发送文件列表时发生未知错误!\n");
        return 0;
 
    }
    return 1;
}
 
 
//发送文件列表信息
//datatcps:数据连接套接字
//返回值:0表示出错,1表示正常
int SendFileList(SOCKET datatcps) {
    HANDLE hff;
    WIN32_FIND_DATA fd;
 
    //搜索文件
    hff = FindFirstFile("*", &fd);
    if (hff == INVALID_HANDLE_VALUE)  //发生错误
    {
        const char* errstr = "不能列出文件!\n";
        printf("文件列表输出失败!\n");
        if (send(datatcps, errstr, strlen(errstr), 0) == SOCKET_ERROR) {
            printf("发送给文件列表时发生未知错误!\n");
        }
        closesocket(datatcps);            return 0;
    }
 
    BOOL fMoreFiles = TRUE;
    while (fMoreFiles) {
        //发送此项文件信息:
        if (!SendFileRecord(datatcps, &fd)) {
            closesocket(datatcps);
            return 0;
        }
        //搜索下一个文件
        fMoreFiles = FindNextFile(hff, &fd);
    }
    closesocket(datatcps);
    return 1;
}
 
//通过数据连接发送文件
int SendFile(SOCKET datatcps, FILE* file) {
    char buf[1024];
    printf("发送文件数据中。。。。。。");
    for (;;) {                //从文件中循环读取数据并发送客户端
        int r = fread(buf, 1, 1024, file);
        if (send(datatcps, buf, r, 0) == SOCKET_ERROR) {
            printf("与客户端失去连接!\n");
            closesocket(datatcps);
            return 0;
        }
        if (r < 1024)   //文件传输结束
        {
            break;
        }
    }
    closesocket(datatcps);
    printf("完成传输!\n");
    return 1;
}
 
//接收文件
//datatcps:数据连接套接字,通过它来接收数据
//filename:用于存放数据的文件名
int RecvFile(SOCKET datatcps, char* filename) {
    char buf[1024];
    FILE* file = fopen(filename, "wb");
    if (!file) {
        printf("写入文件时发生未知错误!\n");
        fclose(file);
        closesocket(datatcps);
        return 0;
    }
    printf("接受文件数据中。。。。。。");
    while (1) {
        int r = recv(datatcps, buf, 1024, 0);
        if (r == SOCKET_ERROR) {
            printf("从客户端接受文件时发生未知错误!\n");
            fclose(file);
            closesocket(datatcps);
            return 0;
        }
        if (!r) {
            break;
        }
        fwrite(buf, 1, r, file);
    }
    fclose(file);
    closesocket(datatcps);
    printf("完成传输!\n");
    return 1;
}
 
//检测文件是否存在:
int FileExists(const char *filename)
{
    WIN32_FIND_DATA fd;
    if (FindFirstFile(filename, &fd) == INVALID_HANDLE_VALUE)
        return 0;
    return 1;
}

3.客户端

头文件:(sizes.h)

#pragma once
 
//服务器侦听控制连接请求的端口
#define CMD_PORT 5858
//客户机侦听数据连接请求的端口
#define DATA_PORT 5850
//命令报文参数缓存的大小
#define CMD_PARAM_SIZE 256
//回复报文消息缓存的大小
#define RSPNS_TEXT_SIZE 256
#define BACKLOG 10
#define DATA_BUFSIZE 4096
 
//命令类型
typedef enum {
    LS, PWD, CD, DOWN, UP, QUIT
} CmdID;
 
//命令报文,从客户端发往服务器
typedef struct _CmdPacket {
    CmdID cmdid;
    char param[CMD_PARAM_SIZE];
} CmdPacket;
 
//回复报文的类型
typedef enum {
    OK, ERR
} RspnsID;
 
//回复报文,从服务器发往客户端
typedef struct _RspnsPacket {
    RspnsID rspnsid;
    char text[RSPNS_TEXT_SIZE];
} RspnsPacket;

源文件:(客户端.cpp)

#include <WinSock2.h>
#include <windows.h>
#include "sizes.h"
#include <tchar.h>
#include <iostream>
 
#pragma comment(lib, "ws2_32.lib")
 
//读取回复报文
void do_read_rspns(SOCKET fd, RspnsPacket *ptr)
{
    int count = 0;
    int size = sizeof(RspnsPacket);
    while (count < size)
    {
        int nRead = recv(fd, (char *)ptr + count, size - count, 0);
        if (nRead <= 0)
        {
            printf("读取服务器的回复失败!\n");
            closesocket(fd);
            exit(1);
        }
        count += nRead;
    }
}
 
//发送命令报文
void do_write_cmd(SOCKET fd, CmdPacket *ptr)
{
    int size = sizeof(CmdPacket);
    int flag = send(fd, (char *)ptr, size, 0);
    if (flag == SOCKET_ERROR)
    {
        printf("给服务器发送命令失败!\n");
        closesocket(fd);
        WSACleanup();
        exit(1);
    }
}
 
//创建数据连接套接字并进入侦听状态
SOCKET create_data_socket()
{
    SOCKET sockfd;
    struct sockaddr_in my_addr;
    //创建用于数据连接的套接字
    if ((sockfd = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP)) == INVALID_SOCKET)
    {
        printf("创建用于数据连接的套接字失败!\n");
        WSACleanup();
        exit(1);
    }
    my_addr.sin_family = AF_INET;
    my_addr.sin_port = htons(DATA_PORT);
    my_addr.sin_addr.s_addr = htonl(INADDR_ANY);
    memset(&(my_addr.sin_zero), 0, sizeof(my_addr.sin_zero));
 
    //绑定
    if (bind(sockfd, (struct sockaddr *)&my_addr, sizeof(struct sockaddr)) == SOCKET_ERROR)
    {
        int err = WSAGetLastError();
        printf("绑定地址失败,错误代码:%d\n", err);
        closesocket(sockfd);
        WSACleanup();
        exit(1);
    }
 
    //侦听数据连接请求
    if (listen(sockfd, 1) == SOCKET_ERROR)
    {
        printf("监听数据连接失败!\n");
        closesocket(sockfd);
        WSACleanup();
        exit(1);
    }
    return sockfd;
}
 
//处理list命令
void list(SOCKET sockfd)
{
    int sin_size;
    int nRead;
    CmdPacket cmd_packet;
    SOCKET newsockfd, data_sockfd;
    struct sockaddr_in their_add;
    char data_buf[DATA_BUFSIZE];
 
    //创建数据连接
    newsockfd = create_data_socket();
    //构建命令报文并发送至服务器
    cmd_packet.cmdid = LS;//没有参数
    do_write_cmd(sockfd, &cmd_packet);
    sin_size = sizeof(struct sockaddr_in);
    //接受服务器的数据连接请求
    if ((data_sockfd = accept(newsockfd, (struct sockaddr*)&their_add, &sin_size)) == INVALID_SOCKET)
    {
        printf("获取文件列表失败!\n");
        closesocket(newsockfd);
        closesocket(sockfd);
        WSACleanup();
        exit(1);
    }
 
    //每次读到多少数据就显示多少,直到数据连接断开
    while (true)
    {
        nRead = recv(data_sockfd, data_buf, DATA_BUFSIZE - 1, 0);
        if (nRead == SOCKET_ERROR)
        {
            printf("读取服务器回复失败!\n");
            closesocket(data_sockfd);
            closesocket(newsockfd);
            closesocket(sockfd);
            WSACleanup();
            exit(1);
        }
 
        if (nRead == 0)//数据读取结束
            break;
 
        //显示数据
        data_buf[nRead] = '\0';
        printf("%s", data_buf);
 
    }
    closesocket(data_sockfd);
    closesocket(newsockfd);
}
//处理pwd命令:
void pwd(int sockfd)
{
    CmdPacket cmd_packet;
    RspnsPacket rspns_packet;
 
    cmd_packet.cmdid = PWD;
    //发送命令报文并读取回复:
    do_write_cmd(sockfd, &cmd_packet);
    do_read_rspns(sockfd, &rspns_packet);
    printf("%s\n", rspns_packet.text);
}
 
//处理cd命令:
void cd(int sockfd)
{
    CmdPacket cmd_packet;
    RspnsPacket rspns_packet;
 
 
    cmd_packet.cmdid = CD;
    scanf("%s", cmd_packet.param);
 
    //发送命令报文并读取回复:
    do_write_cmd(sockfd, &cmd_packet);
    do_read_rspns(sockfd, &rspns_packet);
    if (rspns_packet.rspnsid == ERR)
        printf("%s", rspns_packet.text);
}
 
 
//处理down命令,即下载文件:
void get_file(SOCKET sockfd)
{
    FILE *fd;
    char data_buf[DATA_BUFSIZE];
 
    CmdPacket cmd_packet;
    RspnsPacket rspns_packet;
 
    SOCKET newsockfd, data_sockfd;
    struct sockaddr_in their_addr;
    int sin_size;
    int count;
 
    //设置命令报文:
    cmd_packet.cmdid = DOWN;
    scanf("%s", cmd_packet.param);
 
    //打开或者创建本地文件以供写数据:
    fd = fopen(cmd_packet.param, "wb");//使用二进制方程
    if (fd == NULL)
    {
        printf("打开文件%s来写入失败!\n", cmd_packet.param);
        return;
    }
 
    //创建数据连接并侦听服务器的连接请求:
    newsockfd = create_data_socket();
 
    //发送报文请求:
    do_write_cmd(sockfd, &cmd_packet);
 
    //读取回复报文:
    do_read_rspns(sockfd, &rspns_packet);
    if (rspns_packet.rspnsid == ERR)
    {
        printf("%s", rspns_packet.text);
        closesocket(newsockfd);
 
        fclose(fd);
        //删除文件:
        DeleteFile(cmd_packet.param);
        return;
    }
 
    sin_size = sizeof(struct sockaddr_in);
    //等待接受服务器的连接请求
    if ((data_sockfd = accept(newsockfd, (struct sockaddr *)&their_addr, &sin_size)) == INVALID_SOCKET)
    {
        printf("获取文件失败!\n");
        closesocket(newsockfd);
 
        fclose(fd);
        //删除文件:
        DeleteFile(cmd_packet.param);
        return;
    }
 
    //循环读取网络数据并写入文件:
    while ((count = recv(data_sockfd, data_buf, DATA_BUFSIZE, 0)) > 0)
        fwrite(data_buf, sizeof(char), count, fd);
 
    closesocket(data_sockfd);
    closesocket(newsockfd);
    fclose(fd);
}
 
//处理put命令,即上传文件
void put_file(SOCKET sockfd)
{
    FILE *fd;
    CmdPacket cmd_packet;
    RspnsPacket rspns_packet;
    char data_buf[DATA_BUFSIZE];
 
    SOCKET newsockfd, data_sockfd;
    struct sockaddr_in their_addr;
    int sin_size;
    int count;
    cmd_packet.cmdid = UP;
    scanf("%s", cmd_packet.param);
 
    //打开本地文件用于读取数据
    fd = fopen(cmd_packet.param, "rb");
    if (fd == NULL)
    {
        printf("打开文件%s来读取数据失败!\n", cmd_packet.param);
        return;
    }
 
    //创建数据连接套接字并进入侦听状态;
    newsockfd = create_data_socket();
 
    //发送命令报文
    do_write_cmd(sockfd, &cmd_packet);
 
    //读取回复报文
    do_read_rspns(sockfd, &rspns_packet);
    if (rspns_packet.rspnsid == ERR)
    {
        printf("%s", rspns_packet.text);
        closesocket(newsockfd);
        fclose(fd);
        return;
    }
 
    sin_size = sizeof(struct sockaddr_in);
    //准备接受数据连接
    if ((data_sockfd = accept(newsockfd, (struct sockaddr *)&their_addr, &sin_size)) == INVALID_SOCKET)
    {
        printf("上传文件传输错误!\n");
        closesocket(newsockfd);
        fclose(fd);
        return;
    }
    //循环从文件中读取数据并发给服务器
    while (true)
    {
        count = fread(data_buf, sizeof(char), DATA_BUFSIZE, fd);
        send(data_sockfd, data_buf, count, 0);
        if (count < DATA_BUFSIZE)//数据已经读完或者发生cuowu
            break;
    }
 
    closesocket(data_sockfd);
    closesocket(newsockfd);
    fclose(fd);
}
 
//处理退出命令
void quit(int sockfd)
{
    CmdPacket cmd_packet;
    RspnsPacket rspns_packet;
 
    cmd_packet.cmdid = QUIT;
    do_write_cmd(sockfd, &cmd_packet);
    do_read_rspns(sockfd, &rspns_packet);
    printf("%s", rspns_packet.text);
    getchar();
}
 
void main()
{
    SOCKET sockfd;
    struct sockaddr_in their_addr;
    char cmd[10];
    RspnsPacket rspns_packet;
 
    WORD wVersionRequested;
    WSADATA wsaData;
    int err;
    wVersionRequested = MAKEWORD(2, 2);
    //Winsock初始化
    err = WSAStartup(wVersionRequested, &wsaData);
    if (err != 0)
    {
        printf("WinSock初始化失败!\n");
        return;
    }
 
    //确认WindSock DLL的版本是2.2
    if (LOBYTE(wsaData.wVersion) != 2 || HIBYTE(wsaData.wVersion) != 2)
    {
        printf("WindSock版本不是2.2!\n");
        WSACleanup();
        return;
    }
 
    //创建用于控制谅解的socket
    sockfd = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
    if (sockfd == INVALID_SOCKET)
    {
        printf("创建套接字失败!\n");
        WSACleanup();
        exit(1);
    }
    their_addr.sin_family = AF_INET;
    their_addr.sin_port = htons(CMD_PORT);
    their_addr.sin_addr.S_un.S_addr = inet_addr("127.0.0.1");
    memset(&(their_addr.sin_zero), 0, sizeof(their_addr.sin_zero));
 
    //连接服务器
    if (connect(sockfd, (struct sockaddr*)&their_addr, sizeof(struct sockaddr)) == SOCKET_ERROR)
    {
        printf("连接服务器失败!\n");
        closesocket(sockfd);
        WSACleanup();
        exit(1);
    }
 
    //连接成功后,首先接受服务器发回的消息
    do_read_rspns(sockfd, &rspns_packet);
    printf("%s", rspns_packet.text);
 
    //主循环:读取用户输入并分配执行
    while (true)
    {
        scanf("%s", cmd);
        switch (cmd[0])
        {
        case 'l'://处理List命令
            list(sockfd);
            break;
        case 'p'://处理pwd命令
            pwd(sockfd);
            break;
        case 'c'://处理cd命令
            cd(sockfd);
            break;
        case 'd'://处理down命令
            get_file(sockfd);
            break;
        case 'u'://处理up命令
            put_file(sockfd);
            break;
        case 'q'://处理quit命令
            quit(sockfd);
            break;
        default:
            printf("不存在的命令!\n");
            break;
        }
        if (cmd[0] == 'q')
            break;
    }
    WSACleanup();
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • C++简单QQ程序服务器端的实现代码

    C++简单QQ程序服务器端的实现代码

    这篇文章主要为大家详细介绍了C++简单QQ程序服务器端的实现代码,感兴趣的朋友可以参考一下
    2016-05-05
  • C++ Qt实现音视频播放功能

    C++ Qt实现音视频播放功能

    Qt版本 5.9 基于C++11 Qt核心组件与附加组件安装时请打钩 否则可能出现项目中缺少视频播放模块的问题,由于最近着手的Qt项目需要视频播放自己做的时候踩很多坑避免以后踩坑,故在此记录实现过程,感谢的朋友参考下吧
    2021-11-11
  • C语言解字符串逆序和单向链表逆序问题的代码示例

    C语言解字符串逆序和单向链表逆序问题的代码示例

    这篇文章主要介绍了C语言解字符串逆序和单向链表逆序问题的代码示例,求逆序也是考研和面试中的基础算法题类型,需要的朋友可以参考下
    2016-06-06
  • C++超详细讲解泛型

    C++超详细讲解泛型

    泛型编程,故如其名,是一个泛化的编程方式。其实现原理为程序员编写一个函数/类的代码示例,让编译器去填补出不同的函数实现
    2022-07-07
  • Opencv Hough算法实现图片中直线检测

    Opencv Hough算法实现图片中直线检测

    这篇文章主要为大家详细介绍了Opencv Hough算法实现图片中直线检测,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-12-12
  • C语言数组按协议存储与按协议解析数据的实现

    C语言数组按协议存储与按协议解析数据的实现

    今天小编就为大家分享一篇关于C语言数组按协议存储与按协议解析数据的实现,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2018-12-12
  • C++实现LeetCode(117.每个节点的右向指针之二)

    C++实现LeetCode(117.每个节点的右向指针之二)

    这篇文章主要介绍了C++实现LeetCode(117.每个节点的右向指针之二),本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-07-07
  • 详解C语言中typedef和#define的用法与区别

    详解C语言中typedef和#define的用法与区别

    这篇文章主要给大家介绍了关于C语言中typedef和#define的的用法、区别,以及陷阱。文中通过示例进行了详细讲解,感兴趣的小伙伴可以了解一下
    2022-07-07
  • 解决vscode下调试c/c++程序一闪而过的问题(Windows)

    解决vscode下调试c/c++程序一闪而过的问题(Windows)

    这篇文章主要介绍了解决vscode下调试c/c++程序一闪而过(Windows),本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-08-08
  • C++制作《游戏内存外挂》详解

    C++制作《游戏内存外挂》详解

    这篇文章主要介绍了C++制作《游戏内存外挂》详解,文中通过示例代码和图片介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-07-07

最新评论