使用UDP和套接字进行客户端-服务器文本文件传输 在C++中,服务器创建一个空白文件
Client-server text file transfer using UDP & sockets in C++, server creates a blank file
首先,我想指出我不是一个好的程序员,所以请耐心等待。程序中的逻辑如下:客户端向服务器发送一个文本文件,服务器将其以不同的名称保存。简而言之,非常类似于这个话题:"文件传输服务器/客户端使用套接字",除了我使用不同的协议。我想我已经成功地发送了文件,但似乎服务器,创建一个空白文件后,卡在一个循环/不写任何东西在它。
下面是客户端的代码:
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <netdb.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h> /* memset() */
#include <sys/time.h> /* select() */
#include <errno.h>
#include <sys/stat.h>
#include <fcntl.h>
#define REMOTE_SERVER_PORT 1500
#define MAX_MSG 100
#define PLIK "/home/aatami/Documents/programowanie/serwer/plik.txt"
#define LENGTH 512
int main(int argc, char *argv[]) {
int sd, rc, i,fd;
unsigned int cliLen;
struct sockaddr_in cliAddr, remoteServAddr;
struct hostent *h;
char buf[LENGTH];
/* check command line args */
if(argc<3) {
printf("usage : %s <server> <data1> ... <dataN> n", argv[0]);
exit(1);
}
/* get server IP address (no check if input is IP address or DNS name */
h = gethostbyname(argv[1]);
if(h==NULL) {
printf("%s: unknown host '%s' n", argv[0], argv[1]);
exit(1);
}
printf("%s: sending data to '%s' (IP : %s) n", argv[0], h->h_name,
inet_ntoa(*(struct in_addr *)h->h_addr_list[0]));
remoteServAddr.sin_family = h->h_addrtype;
memcpy((char *) &remoteServAddr.sin_addr.s_addr,
h->h_addr_list[0], h->h_length);
remoteServAddr.sin_port = htons(REMOTE_SERVER_PORT);
/* socket creation */
sd = socket(AF_INET,SOCK_DGRAM,0); /* port 0 - system gets the first free one */
if(sd<0) {
printf("%s: cannot open socket n",argv[0]);
exit(1);
}
/* bind any port */
cliAddr.sin_family = AF_INET;
cliAddr.sin_addr.s_addr = htonl(INADDR_ANY);
cliAddr.sin_port = htons(0);
rc = bind(sd, (struct sockaddr *) &cliAddr, sizeof(cliAddr));
if(rc<0) {
printf("%s: cannot bind portn", argv[0]);
exit(1);
}
/* send data */
for(i=2;i<argc;i++) {
rc = sendto(sd, argv[i], strlen(argv[i])+1, 0,
(struct sockaddr *) &remoteServAddr,
sizeof(remoteServAddr));
if(rc<0) {
printf("%s: cannot send data %d n",argv[0],i-1);
close(sd);
exit(1);
}
/* send text file */
char sdbuf[LENGTH];
printf("[Client] Sending %s to the Server... ", PLIK);
FILE *fs = fopen(PLIK, "r");
if(fs == NULL)
{
printf("ERROR: File %s not found.n", PLIK);
exit(1);
}
bzero(sdbuf, LENGTH);
int fs_block_sz;
while((fs_block_sz = fread(sdbuf, sizeof(char), LENGTH, fs)) > 0)
{
if(send(sd, sdbuf, fs_block_sz, 0) < 0)
{
fprintf(stderr, "ERROR: Failed to send file %s. (errno = %d)n", PLIK, errno);
break;
}
bzero(sdbuf, LENGTH);
}
printf("Ok File %s from Client was Sent!n", PLIK);
}
return 1;
}
服务器的代码:
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <netdb.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h> /* close() */
#include <string.h> /* memset() */
#include <errno.h>
#define LENGTH 512
#define LOCAL_SERVER_PORT 1500
#define MAX_MSG 100
#define PLIKSERV "/home/aatami/Documents/programowanie/serwer/plikserv.txt"
int main(int argc, char *argv[]) { /* licznik argumentow, tablica argumentow */
int sd, rc,nsockfd;
unsigned int n,cliLen;
struct sockaddr_in cliAddr, servAddr;
char msg[MAX_MSG];
char buf[512],sbuf[LENGTH];
/* socket creation */
sd=socket(AF_INET, SOCK_DGRAM, 0);
if(sd<0) {
printf("%s: cannot open socket n",argv[0]);
exit(1);
}
/* bind local server port */
servAddr.sin_family = AF_INET;
servAddr.sin_addr.s_addr = htonl(INADDR_ANY);
servAddr.sin_port = htons(LOCAL_SERVER_PORT);
rc = bind (sd, (struct sockaddr *) &servAddr,sizeof(servAddr));
if(rc<0) {
printf("%s: cannot bind port number %d n",
argv[0], LOCAL_SERVER_PORT);
exit(1);
}
printf("%s: waiting for data on port UDP %un",
argv[0],LOCAL_SERVER_PORT);
/* server infinite loop */
while(1) {
/* init buffer */
memset(msg,0x0,MAX_MSG);
/* receive message */
cliLen = sizeof(cliAddr);
n = recvfrom(sd, msg, MAX_MSG, 0,
(struct sockaddr *) &cliAddr, &cliLen);
if(n<0) {
printf("%s: cannot receive data n",argv[0]);
continue;
}
/* print received message */
printf("%s: from %s:UDP%u : %s n",
argv[0],inet_ntoa(cliAddr.sin_addr),
ntohs(cliAddr.sin_port),msg);
unsigned int sin_size = sizeof(struct sockaddr_in);
FILE *fr = fopen(PLIKSERV, "a");
if(fr == NULL)
printf("File %s Cannot be opened file on server.n", PLIKSERV);
else
{
bzero(sbuf, LENGTH);
int fr_block_sz = 0;
while((fr_block_sz = recvfrom(sd, buf, LENGTH, 0,(struct sockaddr *) &cliAddr, &cliLen)) > 0)
{
int write_sz = fwrite(buf, sizeof(char), fr_block_sz, fr);
if(write_sz < fr_block_sz)
{
perror("File write failed on server.n");
}
bzero(buf, LENGTH);
if (fr_block_sz == 0 || fr_block_sz != 512)
{
break;
}
}
if(fr_block_sz < 0)
{
if (errno == EAGAIN)
{
printf("recv() timed out.n");
}
else
{
fprintf(stderr, "recv() failed due to errno = %dn", errno);
exit(1);
}
}
printf("Ok received from client!n");
fclose(fr);
}
}/* end of server infinite loop */
return 0;
}
任何帮助都非常感谢!
看到你的代码后,我建议尝试从客户端发送一个"Hello worldn"到服务器,看看你是否得到它。一旦在客户端和服务器之间建立了连接,然后尝试扩展代码以执行文件传输。
但是要记住UDP:
"UDP使用一个简单的传输模型和最少的协议机制。它没有握手对话,因此向用户程序暴露了底层网络协议的任何不可靠性。由于这通常是在不可靠的媒体上的IP,因此无法保证交付,订购或重复保护。如果在网络接口级别需要纠错设施,应用程序可以使用为此目的而设计的传输控制协议(TCP)。"
当您启动客户端时,它尽可能快地发送所有数据包,可能是数据包在服务器站点丢失或混乱甚至重复。我还建议等待数据包确认正确的接收。我的意思是,当服务器接收到一个数据包时,它会发送一个数据包向客户端确认它。这样,就可以进行对话(对话),并且可以令人满意地完成传输。
相关文章:
- 如何通过套接字将文本文件的内容从服务器发送到客户端
- 如何从远程 SFTP 服务器获取 HH-MM-SS 时间戳格式的文件列表
- 使用 libssh 从 SFTP 服务器下载文件并使用 C++ 将其写入 ofstream?
- 使用 Libssh 将文件从服务器复制到客户端:分配文件复制目标路径时出现问题
- ofstream将额外的零字节写入Unix服务器上的文件
- 将文件发送到 HTTP 服务器 (C++ Wininet)
- COM 客户端连接 COM 服务器所需的文件是什么(进程外方案)
- 可视化 将 TCP 服务器数据包保存到 C++ 中的文件
- C++应用程序连接到网站服务器并根据密码检索文件
- 为什么我的 Redhat 服务器上的 QuickFIX 进程没有将其核心文件写入应有的位置?
- 通过TCP在客户端/ C#服务器之间传输文件C++
- 从远程服务器请求PDB文件
- 将文件从计算机上传到在线服务器上
- 连接到 FTP 服务器并访问文件
- 在通过套接字向服务器发送文件数据时,我得到了一些垃圾值?为什么
- 为什么文件没有通过 tcp 在 C# 中的客户端和 C 中的服务器之间正确传输(保存?)
- 为什么 tinyxml2 无法解析流量服务器 xml 文件
- 如何在不保存文件的情况下制作打印屏幕并将其发送到FTP服务器?我的工作代码将文件保存到HDD
- 使用SFTP和Pageant认证更新服务器文件夹
- FTP服务器文件传输