目录
server
client
makefile
运行顺序
FTP(File Transfer Protocol)是一种用于在计算机网络上传输文件的标准协议。
它允许用户通过网络将文件从一个计算机(称为FTP服务器)传输到另一个计算机(称为FTP客户端)
以下是FTP的一般工作流程:
- 建立连接:客户端使用FTP客户端软件连接到FTP服务器;
- 身份验证:客户端在连接建立后,提供登录凭据;
- 导航文件系统:验证成功,客户端可以浏览FTP服务器上的文件系统,并选择要上传或下载的文件;
- 上传文件:客户端可以选择要上传到FTP服务器的本地文件,并将其发送到服务器;
- 下载文件:客户端可以选择要从FTP服务器下载的文件,并将其复制到本地计算机;
- 其他操作:FTP还支持其他操作,例如重命名文件、删除文件、创建文件夹等;
项目功能介绍:
均有服务器和客户端代码,基于TCP写的。
在同一路径下,将客户端可执行代码复制到其他的路径下,接下来再不同的路径下运行服务器和客户端。
相当于另外一台电脑在访问服务器。
客户端和服务器链接成功后出现以下提示:四个功能
***************list************** //列出服务器所在目录下的文件名(除目录不显示)
***********put filename********** //上传一个文件
***********get filename********** //重服务器所在路径下载文件
**************quit*************** //退出(可只退出客户端,服务器等待下一个客户端链接)
server
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/ip.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/types.h>
#include <dirent.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>void list_server(int acceptfd, char *buf, int size); //list功能
void put_server(int acceptfd, char *buf, int size); //put功能
void get_server(int acceptfd,char *buf, int size); //get功能
int main(int argc, char const *argv[])
{if (argc != 2){printf("please input %s <port>\n", argv[0]);return -1;}//1.创建流式套接字int sockfd = socket(AF_INET, SOCK_STREAM, 0); //链接if (sockfd < 0){perror("socket err.");return -1;}printf("sockfd:%d\n", sockfd); //3//填充ipv4的通信结构体struct sockaddr_in saddr, caddr;saddr.sin_family = AF_INET;saddr.sin_port = htons(atoi(argv[1])); //"8888" int a= atoi("8888")//a=8888saddr.sin_addr.s_addr = inet_addr("0.0.0.0");socklen_t len = sizeof(caddr);//2.绑定套接字 ip和端口(自己)if (bind(sockfd, (struct sockaddr *)&saddr, sizeof(saddr)) < 0){perror("bind err.");return -1;}printf("bind ok.\n");//3.监听if (listen(sockfd, 5) < 0){perror("listen err.");return -1;}printf("listen ok.\n");//4.阻塞等待客户端链接while (1){int acceptfd = accept(sockfd, (struct sockaddr *)&caddr, &len);if (acceptfd < 0){perror("accept err.");return -1;}printf("acceptfd=%d\n", acceptfd); //通信printf("client:ip=%s port=%d\n",inet_ntoa(caddr.sin_addr),ntohs(caddr.sin_port));//循环收发消息char buf[128];int ret;while (1){ret = recv(acceptfd, buf, sizeof(buf), 0);if (ret < 0){perror("recv err.");return -1;}else if (ret == 0){printf("client exit\n");break;}else{if (strncmp(buf, "list", 4) == 0){ //打开当前目录读文件判断文件是普通文件将文件名传给客户端list_server(acceptfd, buf, sizeof(buf));}else if (strncmp(buf, "put ", 4) == 0){ //接收文件(本地:打开新建文件接收客户端发送过来的内容写到文件)put_server(acceptfd, buf, sizeof(buf));}else if (strncmp(buf, "get ", 4) == 0){ //发送文件//(本地:打开文件读文件内容发送给客户端)get_server(acceptfd, buf, sizeof(buf));}}}close(acceptfd);}close(sockfd);return 0;
}//3、get
void get_server(int acceptfd,char *buf, int size) //buf->put xxx.c
{//1>打开文件 读int fd = open(buf + 4, O_RDONLY); //buf->put xxx.cif (fd < 0){perror("open file err.");return;}int ret; //实际读到的个数while ((ret = read(fd, buf, size - 1)) != 0){buf[ret] = '\0'; // 将读取的内容以字符串形式结束send(acceptfd, buf, size, 0); // 发送读取的内容给客户端}strcpy(buf, "get ok."); // 将"get ok."字符串复制到缓冲区send(acceptfd, buf, size, 0); // 发送"get ok."给客户端
}
//2.接收文件(本地:打开新建文件接收客户端发送过来的内容写到文件)
void put_server(int acceptfd, char *buf, int size)
{// 1. 打开文件,以写入方式打开(如果不存在则创建,如果存在则清空内容)int fd = open(buf + 4, O_TRUNC | O_CREAT | O_WRONLY, 0666);if (fd < 0){perror("open err.");return;}while (1){// 2. 接收客户端发送的数据if (recv(acceptfd, buf, size, 0) < 0){perror("recv err.");return;}// 3. 检查是否接收到结束标识"put ok."if (strncmp(buf, "put ok.", 7) == 0)break;// 4. 将接收到的数据写入文件write(fd, buf, strlen(buf));}
}//list:打开当前目录读文件判断文件是普通文件将文件名传给客户端
void list_server(int acceptfd, char *buf, int size)
{//1.打开当前目录文件DIR *dir = opendir("./");if (dir == NULL){perror("opendir err.");return;}//2.循环读目录文件 readdirstruct dirent *dp = NULL;struct stat st;while ((dp = readdir(dir)) != NULL){//dp->d_name拿到的文件名//判断文件属性statstat(dp->d_name, &st);if (S_ISREG(st.st_mode)){strcpy(buf, dp->d_name); // 将文件名拷贝到缓冲区 send(acceptfd, buf, size, 0); // 发送文件名给客户端}}//发送结束标志strcpy(buf, "list ok.");send(acceptfd, buf, size, 0);
}
client
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/ip.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <string.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>void show(void);
void list_client(int sockfd, char *buf, int size);
void put_client(int sockfd, char *buf, int size);
void get_client(int sockfd, char *buf, int size);
int main(int argc, char const *argv[])
{if (argc != 3){printf("please input %s <ip> <port>\n", argv[0]);return -1;}//1.创建套接子int sockfd = socket(AF_INET, SOCK_STREAM, 0);if (sockfd < 0){perror("socket err.");return -1;}//填充结构体struct sockaddr_in saddr;saddr.sin_family = AF_INET;saddr.sin_port = htons(atoi(argv[2]));saddr.sin_addr.s_addr = inet_addr(argv[1]);if (connect(sockfd, (struct sockaddr *)&saddr, sizeof(saddr)) < 0){perror("connect err.");return -1;}//收发消息char buf[128];while (1){//1.请求窗口show();//2.获取请求fgets(buf, sizeof(buf), stdin);if (buf[strlen(buf) - 1] == '\n')buf[strlen(buf) - 1] = '\0';send(sockfd, buf, sizeof(buf), 0);if (strncmp(buf, "list", 4) == 0){ //循环接收服务器发送过来的普通文件名输出到终端list_client(sockfd, buf, sizeof(buf));}else if (strncmp(buf, "put ", 4) == 0){ //上传文件(本地:打开文件读内容发送给服务器)put_client(sockfd,buf,sizeof(buf));}else if (strncmp(buf, "get ", 4) == 0){ //下载服务器路径下的文件//(本地:打开新建文件接收服务器发送过来的内容写到文件)get_client(sockfd,buf,sizeof(buf));}else if (strncmp(buf, "quit", 4) == 0){break;}}close(sockfd);return 0;
}
//3.get
void get_client(int sockfd, char *buf, int size)
{int fd = open(buf + 4, O_TRUNC | O_CREAT | O_WRONLY, 0666);if (fd < 0){perror("open err.");return;}while (1){if (recv(sockfd, buf, size, 0) < 0){perror("recv err.");return;}if (strncmp(buf, "get ok.", 7) == 0)break;write(fd, buf, strlen(buf));}
}
//2.put 上传文件(本地:打开文件读内容发送给服务器)
void put_client(int sockfd, char *buf, int size) //buf->put xxx.c
{//1>打开文件 读int fd = open(buf + 4, O_RDONLY);if (fd < 0){perror("open file err.");return;}int ret; //实际读到的个数while ((ret = read(fd, buf, size - 1)) != 0){buf[ret] = '\0';send(sockfd, buf, size, 0);}strcpy(buf, "put ok.");send(sockfd, buf, size, 0);
}//1.list:循环接收服务器发送过来的普通文件名输出到终端
void list_client(int sockfd, char *buf, int size)
{while (1){if (recv(sockfd, buf, size, 0) < 0){perror("list recv err.");return;}if (strncmp(buf, "list ok.", 8) == 0){break;}printf("%s\n", buf);}
}void show(void)
{printf("************list***************\n");printf("************put filename*******\n");printf("************get filename*******\n");printf("************quit***************\n");
}
makefile
all:gcc server.c -o ser gcc client.c -o cli
.PHONY:clean
clean:rm ser cli