利用TCP编程实现FTP功能

模拟FTP核心原理:客户端连接服务器后,向服务器发送一个文件。文件名可以通过参数指定,服务器端接收客户端传来的文件(文件名随意),如果文件不存在自动创建文件,如果文件存在,那么清空文件然后写入。
项目功能介绍:
均有服务器和客户端代码,基于TCP 写的。
在同一路径下,将客户端可执行代码复制到其他的路径下,接下来在不同的路径下运行服务器和客户端。相当于另外一台电脑在访问服务器。
客户端和服务器链接成功后出现以下提示:四个功能
***************list**************//列出服务器所在目录下的普通文件名
***********put filename**********//从客户端所在路径上传文件
***********get filename**********//从服务器所在路径下载文件
**************quit***************//退出(可只退出客户端,服务器等待下一个客户端链接)

思路:
list:客户端输入list------》把list发送给服务器-------》接收list---》判断是否为list----》目录操作(循环读目录文件)--------》判断是否为普通文件-----》如果是普通文件就发送给客户端------》发送一个结束"end"的标志----------》客户端循环接收普通文件并打印到终端显示
put filename:客户端输入put 文件名---》把put 文件名发送给服务器---》接收put 文件名---》判断是否为put -----》(cp:源文件再客户端所在路径下,目标文件在服务器所在路径下)客户端循环读源文件发送给服务器,服务器循环接收写到目标文件里--------》发送一个结束"end"的标志

接收大小与发送大小一致(避免沾包)

客户端:

#include <stdio.h>
#include <arpa/inet.h>
#include <string.h>
#include <dirent.h>
#include <sys/stat.h>
#include <unistd.h>
#include <sys/types.h>
#include <fcntl.h>
#include <string.h>
#include <stdlib.h>
void show()
{
    printf("***************list**************\n");
    printf("***********put filename**********\n");
    printf("***********get filename**********\n");
    printf("**************quit***************\n");
}
int list(int acceptfd)
{
    char buf[128];
    int ret;
    while (1)
    {
        ret = recv(acceptfd, buf, sizeof(buf), 0);
        if (ret < 0)
        {
            printf("recv err\n");
            return -1;
        }
        else if (ret == 0)
        {
            printf("client exit\n");
            return -1;
        }
        if (!strcmp(buf, "end"))
        {
            printf("\n");
            break;
        }
        printf("%s ", buf);
    }
}
int put(int sockfd, char *p)
{
    char buf[128];
    int fd = open(p + 4, O_RDONLY);
    if (fd < 0)
    {
        printf("open err\n");
        return -1;
    }
    while (read(fd, buf, sizeof(buf)-1))
    {
        send(sockfd, buf, sizeof(buf), 0);
        memset(buf, 0, sizeof(buf));
    }
    strcpy(buf, "over");
    send(sockfd, buf, sizeof(buf), 0);
    close(fd);
}
int get(int sockfd, char *p)
{
    char buf[128]={0};
    int fd = open(p + 4, O_WRONLY | O_CREAT | O_TRUNC ,0777);
    if (fd < 0)
    {
        printf("get open err\n");
        return -1;
    }
    while (1)
    {
        int ret = recv(sockfd, buf, sizeof(buf), 0);
        if (ret < 0)
        {
            perror("recv err");
            return -1;
        }
        else if (ret == 0)
        {
            printf("client exit\n");
            break;
        }
        if (!strcmp(buf, "over"))
            break;
        write(fd, buf, strlen(buf));
        memset(buf, 0, sizeof(buf));
    }
    close(fd);
}
int main(int argc, char const *argv[])
{
    char buf[128] = {0};
    // 1.创建套接字(socket)------------》有手机
    int sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if (sockfd < 0)
    {
        perror("socket err");
        return -1;
    }
    printf("sockfd:%d\n", sockfd);
    // 2.指定(服务器)网络信息--------》有对方的号码
    struct sockaddr_in saddr;
    saddr.sin_family = AF_INET;
    saddr.sin_port = htons(atoi(argv[1]));
    saddr.sin_addr.s_addr = inet_addr("192.168.50.241"); 
    //saddr.sin_addr.s_addr = INADDR_ANY;
    // 3.连接(connect)-------------------》拨打电话
    if (connect(sockfd, (struct sockaddr *)&saddr, sizeof(saddr)) < 0)
    {
        perror("connect err");
        return -1;
    }
    printf("connect okk\n");
    // 4.接收发送消息(recv send)---》通话
    show();
    while (1)
    {
        fgets(buf, sizeof(buf), stdin);
        if (buf[strlen(buf) - 1] == '\n')
            buf[strlen(buf) - 1] = '\0';
        send(sockfd, buf, sizeof(buf), 0);
        if (!strcmp(buf, "quit"))
            break;
        else if (!strcmp(buf, "list"))
            list(sockfd);
        else if (!strncmp(buf, "put ", 4))
            put(sockfd, buf);
        else if (!strncmp(buf, "get ", 4))
            get(sockfd, buf);
        else
            printf("命令无效,请重新输入\n");
        memset(buf,0, sizeof(buf));
        // write(sockfd, buf, sizeof(buf));
    }
    // 5.关闭套接字(close)------------》挂电话
    close(sockfd);
    return 0;
}

服务器:

#include <stdio.h>
#include <arpa/inet.h>
#include <string.h>
#include <dirent.h>
#include <sys/stat.h>
#include <unistd.h>
#include <sys/types.h>
#include <fcntl.h>
#include <string.h>
#include <stdlib.h>
int list(int acceptfd)
{
    char buf[128];
    DIR *dr = opendir(".");
    if (dr == NULL)
    {
        printf("opendir err\n");
        return -1;
    }
    struct dirent *dir;
    while (dir = readdir(dr))
    {
        struct stat dir1;
        stat(dir->d_name, &dir1);
        if ((dir1.st_mode & __S_IFMT) == __S_IFREG)
        {
            strcpy(buf, dir->d_name);
            send(acceptfd, buf, sizeof(buf), 0);
            memset(buf, 0, sizeof(buf));
        }
    }
    strcpy(buf, "end");
    send(acceptfd, buf, sizeof(buf), 0);
    closedir(dr);
}
int get(int acceptfd, char *p)
{
    char buf[128];
    int fd = open(p + 4, O_RDONLY);
    if (fd < 0)
    {
        printf("open err\n");
        return -1;
    }
    while (read(fd, buf, sizeof(buf)))
    {
        send(acceptfd, buf, sizeof(buf), 0);
        memset(buf, 0, sizeof(buf));
    }
    strcpy(buf, "over");
    send(acceptfd, buf, sizeof(buf), 0);
    close(fd);
}
int put(int acceptfd, char *p)
{
    char buf[128] = {0};
    int fd = open(p + 4, O_WRONLY | O_CREAT | O_TRUNC, 0777);
    if (fd < 0)
    {
        printf("get open err\n");
        return -1;
    }
    while (1)
    {
        int ret = recv(acceptfd, buf, sizeof(buf), 0);
        if (ret < 0)
        {
            perror("recv err");
            return -1;
        }
        else if (ret == 0)
        {
            printf("client exit\n");
            break;
        }
        if (!strcmp(buf, "over"))
            break;
        write(fd, buf, strlen(buf));
        memset(buf, 0, sizeof(buf));
    }
    close(fd);
}
int main(int argc, char const *argv[])
{
    char buf[128] = {0};
    int ret, acceptfd;
    // 1.创建套接字(socket)---------------》有手机
    int sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if (sockfd < 0)
    {
        perror("socket err");
        return -1;
    }
    printf("sockfd:%d\n", sockfd); // 3
    // 2.指定网络信息---------------------------》有号码
    struct sockaddr_in saddr, caddr;
    saddr.sin_family = AF_INET;            // IPV4
    saddr.sin_port = htons(atoi(argv[1])); // 端口号
    saddr.sin_addr.s_addr = inet_addr("192.168.50.241"); // 虚拟机IP
    // saddr.sin_addr.s_addr = inet_addr("0.0.0.0");
    //saddr.sin_addr.s_addr = INADDR_ANY;
    int len = sizeof(caddr);
    // 3.绑定套接字(bind)------------------》绑定手机(插卡)
    if (bind(sockfd, (struct sockaddr *)&saddr, sizeof(saddr)) < 0)
    {
        perror("bind err");
        return -1;
    }
    printf("bind ok\n");
    // 4.监听套接字(listen)-----------------》待机
    if (listen(sockfd, 6) < 0)
    {
        perror("listen err");
        return -1;
    }
    printf("listen ok\n");
    // 5.接收客户端连接连接请求(accept)--》接电话
    // tcp服务器一共有两类文件描述符,一类用于连接,一类用于通信
    // socket函数返回值:用于连接的文件描述符
    // accept函数返回值:用于通信的文件描述符
    while (1)
    {

        acceptfd = accept(sockfd, (struct sockaddr *)&caddr, &len);
        if (acceptfd < 0)
        {
            perror("accept err");
            return -1;
        }
        printf("port:%d ip:%s\n", ntohs(caddr.sin_port), inet_ntoa(caddr.sin_addr));
        printf("acceptfd:%d\n", acceptfd);

        // 6.接收、发送数据(recv send)---》通话
        while (1)
        {
            // read/write()
            ret = recv(acceptfd, buf, sizeof(buf), 0);
            if (ret < 0)
            {
                perror("recv err");
                return -1;
            }
            else if (ret == 0)
            {
                printf("client exit\n");
                break;
            }
            if (!strcmp(buf, "list"))
                list(acceptfd);
            else if (!strncmp(buf, "get ", 4))
                put(acceptfd, buf);
            else if (!strncmp(buf, "put ", 4))
                put(acceptfd, buf);
        }
        close(acceptfd);
    }
    close(sockfd);
    return 0;
}
相关推荐
qq_41656020几秒前
fmql之ubuntu添加dhcp服务
网络
小狮子安度因35 分钟前
边缘智能-大模型架构初探
linux·网络
宇宙第一小趴菜1 小时前
探索网络世界:TCP/IP协议、Python Socket编程及日常生活比喻
网络·python·tcp/ip
新手嵌入式学习1 小时前
Modbus_tcp
网络·网络协议·tcp/ip
小宋10212 小时前
RabbitMQ:交换机详解(Fanout交换机、Direct交换机、Topic交换机)
服务器·网络·rabbitmq
hellojackjiang20112 小时前
开源即时通讯IM框架MobileIMSDK的H5端技术概览
网络·开源·即时通讯·im开发
yczykjyxgs2 小时前
PCDN技术如何保障网络安全?(贰)
服务器·网络·科技
StevenZeng学堂3 小时前
【Kubernetes笔记】为什么DNS解析会超时?
网络协议·docker·云原生·kubernetes
SchneeDuan4 小时前
HTTP和HTTPS的区别
网络协议·http·https·ssl/tls协议·数字证书、签名
伪NChris4 小时前
机器学习的网络们
网络·人工智能·机器学习