当前位置:首页 » 《关注互联网》 » 正文

利用TCP编程实现FTP功能

20 人参与  2024年10月11日 11:21  分类 : 《关注互联网》  评论

点击全文阅读


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

点击全文阅读


本文链接:http://zhangshiyu.com/post/170452.html

<< 上一篇 下一篇 >>

  • 评论(0)
  • 赞助本站

◎欢迎参与讨论,请在这里发表您的看法、交流您的观点。

关于我们 | 我要投稿 | 免责申明

Copyright © 2020-2022 ZhangShiYu.com Rights Reserved.豫ICP备2022013469号-1