gpt4 book ai didi

c - 在 linux 上使用 c 中的套接字将文件从客户端发送到服务器

转载 作者:太空宇宙 更新时间:2023-11-04 12:38:54 25 4
gpt4 key购买 nike

我对 C 语言和 linux 很陌生,英语不是我的母语。对不起那些提前。我需要做的是在 Linux 上使用套接字将 .avi.mp4 文件从客户端发送到服务器。我可以将文件从客户端发送到服务器,但服务器上的视频不工作。

当我尝试播放视频时,我不断收到“无法确定流的类型”之类的错误。当我在客户端查看原始视频时,大小为 5,787,969 字节,但在服务器上传输的视频大小为 5,786,954 字节。我认为这是因为我在传输文件时丢失了数据。

我该如何解决这个问题?

下面是我的代码:

服务器

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h> // read, write
#include <arpa/inet.h>
#include <sys/types.h> // socket, bind, accept, open
#include <sys/socket.h> // socket, bind, listen, accept
#include <sys/stat.h> // open
#include <fcntl.h> // open
#include <errno.h>

#define PORT 5500
#define MAXBUF 1024

int main() {
int server_sockfd;
int client_sockfd;
int des_fd; // file num
struct sockaddr_in serveraddr, clientaddr;
int client_len, read_len, file_read_len; // length
char buf[MAXBUF];

int check_bind;
client_len = sizeof(clientaddr);

/* socket() */
server_sockfd = socket(AF_INET, SOCK_STREAM, 0);
if(server_sockfd == -1) {
perror("socket error : ");
exit(0);
}

/* bind() */
bzero(&serveraddr, sizeof(serveraddr));
serveraddr.sin_family = AF_INET;
serveraddr.sin_addr.s_addr = htonl(INADDR_ANY);
serveraddr.sin_port = htons(PORT);

if(bind(server_sockfd, (struct sockaddr *)&serveraddr, sizeof(serveraddr)) > 0) {
perror("bind error : ");
exit(0);
}

/* listen */
if(listen(server_sockfd, 5) != 0) {
perror("listen error : ");
}

while(1) {
char file_name[MAXBUF]; // local val
memset(buf, 0x00, MAXBUF);

/* accept() */
client_sockfd = accept(server_sockfd, (struct sockaddr *)&clientaddr, &client_len);
printf("New Client Connect : %s\n", inet_ntoa(clientaddr.sin_addr));

/* file name */
read_len = read(client_sockfd, buf, MAXBUF);
if(read_len > 0) {
strcpy(file_name, buf);
printf("%s > %s\n", inet_ntoa(clientaddr.sin_addr), file_name);
} else {
close(client_sockfd);
break;
}

/* create file */

des_fd = open(file_name, O_WRONLY | O_CREAT | O_EXCL, 0700);
if(!des_fd) {
perror("file open error : ");
break;
}
/* file save */
while(1) {
memset(buf, 0x00, MAXBUF);
file_read_len = read(client_sockfd, buf, MAXBUF);
write(des_fd, buf, file_read_len);
if(file_read_len == EOF | file_read_len == 0) {
printf("finish file\n");
break;
}


}


close(client_sockfd);
close(des_fd);
}
close(server_sockfd);
return 0;
}

客户端

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <unistd.h>
#include <sys/stat.h>
#include <fcntl.h>

#define PORT 5500
#define IP "127.0.0.1"
#define MAXBUF 1024

int main() {
struct sockaddr_in serv_addr;
int s;
int sourse_fd;
char buf[MAXBUF];
int file_name_len, read_len;
/* socket() */
s = socket(AF_INET, SOCK_STREAM, 0);
if(s == -1) {
return 1;
}
serv_addr.sin_family = AF_INET;
serv_addr.sin_addr.s_addr = inet_addr(IP);
serv_addr.sin_port = htons(PORT);

if(connect(s, (struct sockaddr *)&serv_addr, sizeof(serv_addr)) == -1) {
perror("connect : ");
printf("fail to connect.\n");
close(s);
return 1;
}

memset(buf, 0x00, MAXBUF);
printf("write file name to send to the server: ");
scanf("%s", buf);

printf(" > %s\n", buf);
file_name_len = strlen(buf);

send(s, buf, file_name_len, 0);
sourse_fd = open(buf, O_RDONLY);
if(!sourse_fd) {
perror("Error : ");
return 1;
}

while(1) {
memset(buf, 0x00, MAXBUF);
read_len = read(sourse_fd, buf, MAXBUF);
send(s, buf, read_len, 0);
if(read_len == 0) {
break;
}

}

return 0;
}

最佳答案

在服务器中

Server 中循环查看您的“文件保存”:

/* file save */
while(1) {
memset(buf, 0x00, MAXBUF);
file_read_len = read(client_sockfd, buf, MAXBUF);
write(des_fd, buf, file_read_len);
if(file_read_len == EOF | file_read_len == 0) {
printf("finish file\n");
break;
}
}

代替 file_read_len == EOF | file_read_len == 0 你应该只使用 file_read_len == 0

仍在 Server 中,在创建文件的 open() 函数中,我还建议将 0700 更改为 S_IRWXU & (~S_IXUSR ) __but_it_is_not_mandatory__。有关详细信息,请参阅 user@host:~$ man 2 open


在客户端

Client中你只需要在strlen(buf)中加上“+1”:
file_name_len = strlen(buf) + 1;
那是因为字符串需要以空字节结尾。 strlen() 返回字符串的长度减去末尾的空字节:-)

我还建议你使用argcargv 来获取参数,看:

#include <fcntl.h>
#include <stdio.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>

int
main (int argc, char **argv) {
int how_many_files = argc - 1; /* '- 1' cause first arg is program name
*/
int fd; /* file descriptor */
int i; /* a counter */
char buffer[10]; /* a buffer */

for (i = 0; i < how_many_files; i++) {
fd = open(argv[i + 1], O_RDONLY);
if (fd == -1)
continue;

read(fd, buffer, 10);
buffer[9] = '\0'; /* last byte of buffer needs to be null char*/
printf("First 10 Bytes of the file is: %s\n",
buffer);

close(fd);
}

return 0;
}

关于c - 在 linux 上使用 c 中的套接字将文件从客户端发送到服务器,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40786888/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com