用socket实现简单的文件传输

http://atu82.bokee.com/4667560.html

服务器端:
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
#include <netdb.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <sys/socket.h>
#define SERVPORT 6666
#define MAXDATASIZE 1024 /*每次最大数据传输量 */
#define FILETOWRITE "/home/xufeng/xf_write"
#define BACKLOG 10

int main()
{
 FILE *fpw;
 int sockfd,client_fd;
 int recvbytes;
 struct sockaddr_in my_addr; /* 本机地址信息 */
 struct sockaddr_in remote_addr; /* 客户端地址信息 */
 unsigned char buf[MAXDATASIZE];
 char control[4]="YES";
 memset(buf, '\0', sizeof(buf));
 

 if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) == -1)
 { 
  perror("socket initialization Error.");
  exit(1);
 }
 
 my_addr.sin_family=AF_INET;   
 my_addr.sin_port=htons(SERVPORT); 
 my_addr.sin_addr.s_addr = INADDR_ANY; 
 bzero(&(my_addr.sin_zero),8); 

 if (bind(sockfd, (struct sockaddr *)&my_addr, sizeof(struct sockaddr)) == -1)
 {
  perror("bind Error");
  exit(1);
 }
 
 if (listen(sockfd, BACKLOG) == -1)
 {
  perror("listen Error!");
  exit(1);
 }
 
 while(1)
 {
  int sin_size = sizeof(struct sockaddr_in);
  if ((client_fd = accept(sockfd, (struct sockaddr *)&remote_addr, &sin_size)) == -1)
  {
   perror("accept Error");
   continue;
  }
   
  printf("received a connection from %s\n", inet_ntoa(remote_addr.sin_addr)); //inet_ntoa(将网络二进制的数字转换成网络地址)
  
  if ((fpw=fopen(FILETOWRITE, "ab+")) == NULL)
  {
   printf("File to Write Open error.\n");
   exit(1);
  }
  if (send(client_fd, control,strlen(control), 0) == -1)
  {
   perror("send Error");
   /* 这个通讯已经结束     */
   exit(0);
  }
  printf("Send YES: clear to receive data\n");
  while(1)
  {
   if ((recvbytes=recv(client_fd, buf,MAXDATASIZE, 0)) ==-1)
   {
    perror("recv Error!");
    exit(1);
   }
   buf[recvbytes] = '\0';
   //printf("Received:\n%s\nlength: %d\n",buf, strlen(buf));

   // open file to write
   //printf("begin writing...\n");
   fwrite(buf, 1, recvbytes, fpw);
   memset(buf, '\0', sizeof(buf));
   if(recvbytes < 1000)
    break;
   if (send(client_fd, control,strlen(control), 0) == -1)
   {
    perror("send Error");
    /* 这个通讯已经结束     */
    exit(0);
   }
   printf("Send YES: clear to receive data\n");
  }
  //printf("end writing\n");
  /* 循环下一个     */  
  close(client_fd);
  fclose(fpw);
 }
 
}

客户端:
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
#include <netdb.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <sys/socket.h>

#define SERVPORT 6666
#define MAXDATASIZE 1024 /*每次最大数据传输量 */
#define FILETOREAD "/home/xufeng/xf_read"

int main()
{
 FILE *fpr;
 int i=0;
 unsigned char buf[MAXDATASIZE];
 int readlength;
 int sockfd;
 struct sockaddr_in serv_addr;
 int recvbytes;

 memset(buf, '\0', sizeof(buf));
 if ((fpr=fopen(FILETOREAD, "rb")) == NULL)
 {
  printf("File to Read Open error.\n");
  exit(1);
 } 
 
 if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) == -1)
 {
  perror("socket initialization failure.\n");
  exit(1);
 }
 
 memset(&serv_addr, 0, sizeof(serv_addr));
 serv_addr.sin_family=AF_INET;
 serv_addr.sin_port=htons(SERVPORT);
 serv_addr.sin_addr.s_addr = inet_addr("127.0.0.1");
 bzero(&(serv_addr.sin_zero),8);

 if (connect(sockfd, (struct sockaddr *)&serv_addr, sizeof(struct sockaddr)) == -1)
 {
  perror("connect Error!");
  exit(1);
 }

 while(1)
 {
  if ((recvbytes=recv(sockfd, buf, 3, 0)) ==-1)
  {
   perror("recv Error!");
   exit(1);
  }
  buf[recvbytes]='\0';
  printf("receive control signal: %s\n", buf);
  if(strcmp(buf, "YES")==0)
  {
   readlength = fread(buf, 1, 1000, fpr); 
   buf[readlength]='\0';
   //printf("send:\n%s\nlength: %d\n", buf, strlen(buf));
   if(readlength < 1000)
    break;
   if (send(sockfd, buf, readlength, 0) == -1)
   {
    perror("send error\n");
    exit(1);
   }
   memset(buf, '\0', sizeof(buf));
  }
 }
 //fread(buf, 1, readlength, fpr);
 if (send(sockfd, buf, readlength, 0) == -1)
 {
  perror("send error\n");
  exit(1);
 }
 //printf("send:\n%s\nlength: %d\n", buf, strlen(buf));
 memset(buf, '\0', sizeof(buf));
 close(sockfd);
 fclose(fpr);
}

这是我在学习socket的时候编写的简单的文件上传程序,在本机上实现模拟文件传输.
原理是这样的:客户端打开一个文件,每次读1000个字节,然后通过socket传送到服务器,服务器接受字节流,创建一个新文件,再写文件.
写程序的过程中遇到一个问题:当传送小文件的时候没有什么事情,但是当传送较大的文件的时候,就会得到无法预料的结果,有可能造成socket pipe broken(socket管道破裂),推测原因,可能是由于文件读的速度远远快于文件写的速度.后来我加了一段简单的流量控制程序,类似于停止等待协议,即发送方发完数据后等待接受方传送过来的控制信令(本程序采用了一个字符串"YES")代表发送方可以继续发送字节流.这种方法可以解决不能传送大文件的问题,只是采用类似停止等待协议的思想本身效率可能不高.

不知道FTP协议采用的哪种流量控制方法?希望有人指点.

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值