当前位置:  编程技术>c/c++/嵌入式

Linux网络编程之基于UDP实现可靠的文件传输示例

    来源: 互联网  发布时间:2014-10-27

    本文导语:  了解网络传输协议的人都知道,采用TCP实现文件传输很简单。相对于TCP,由于UDP是面向无连接、不可靠的传输协议,所以我们需要考虑丢包和后发先至(包的顺序)的问题,所以我们想要实现UDP传输文件,则需要解决这两个问...

了解网络传输协议的人都知道,采用TCP实现文件传输很简单。相对于TCP,由于UDP是面向无连接、不可靠的传输协议,所以我们需要考虑丢包和后发先至(包的顺序)的问题,所以我们想要实现UDP传输文件,则需要解决这两个问题。方法就是给数据包编号,按照包的顺序接收并存储,接收端接收到数据包后发送确认信息给发送端,发送端接收确认数据以后再继续发送下一个包,如果接收端收到的数据包的编号不是期望的编号,则要求发送端重新发送。

下面展示的是基于linux下C语言实现的一个示例程序,该程序定义一个包的结构体,其中包含数据和包头,包头里包含有包的编号和数据大小,经过测试后,该程序可以成功传输一个视频文件。

具体实现代码如下:

server端代码如下:

/************************************************************************* 
  > File Name: server.c 
  > Author: SongLee 
 ************************************************************************/ 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
 
#define SERVER_PORT 8000 
#define BUFFER_SIZE 1024 
#define FILE_NAME_MAX_SIZE 512 
 
/* 包头 */ 
typedef struct 
{ 
  int id; 
  int buf_size; 
}PackInfo; 
 
/* 接收包 */ 
struct SendPack 
{ 
  PackInfo head; 
  char buf[BUFFER_SIZE]; 
} data; 
 
 
int main() 
{ 
  /* 发送id */ 
  int send_id = 0; 
 
  /* 接收id */ 
  int receive_id = 0; 
 
  /* 创建UDP套接口 */ 
  struct sockaddr_in server_addr; 
  bzero(&server_addr, sizeof(server_addr)); 
  server_addr.sin_family = AF_INET; 
  server_addr.sin_addr.s_addr = htonl(INADDR_ANY); 
  server_addr.sin_port = htons(SERVER_PORT); 
 
  /* 创建socket */ 
  int server_socket_fd = socket(AF_INET, SOCK_DGRAM, 0); 
  if(server_socket_fd == -1) 
  { 
    perror("Create Socket Failed:"); 
    exit(1); 
  } 
 
  /* 绑定套接口 */ 
  if(-1 == (bind(server_socket_fd,(struct sockaddr*)&server_addr,sizeof(server_addr)))) 
  { 
    perror("Server Bind Failed:"); 
    exit(1); 
  } 
 
  /* 数据传输 */ 
  while(1) 
  {   
    /* 定义一个地址,用于捕获客户端地址 */ 
    struct sockaddr_in client_addr; 
    socklen_t client_addr_length = sizeof(client_addr); 
 
    /* 接收数据 */ 
    char buffer[BUFFER_SIZE]; 
    bzero(buffer, BUFFER_SIZE); 
    if(recvfrom(server_socket_fd, buffer, BUFFER_SIZE,0,(struct sockaddr*)&client_addr, &client_addr_length) == -1) 
    { 
      perror("Receive Data Failed:"); 
      exit(1); 
    } 
 
    /* 从buffer中拷贝出file_name */ 
    char file_name[FILE_NAME_MAX_SIZE+1]; 
    bzero(file_name,FILE_NAME_MAX_SIZE+1); 
    strncpy(file_name, buffer, strlen(buffer)>FILE_NAME_MAX_SIZE?FILE_NAME_MAX_SIZE:strlen(buffer)); 
    printf("%sn", file_name); 
 
    /* 打开文件 */ 
    FILE *fp = fopen(file_name, "r"); 
    if(NULL == fp) 
    { 
      printf("File:%s Not Found.n", file_name); 
    } 
    else 
    { 
      int len = 0; 
      /* 每读取一段数据,便将其发给客户端 */ 
      while(1) 
      { 
        PackInfo pack_info; 
 
        if(receive_id == send_id) 
        { 
          ++send_id; 
          if((len = fread(data.buf, sizeof(char), BUFFER_SIZE, fp)) > 0) 
          { 
            data.head.id = send_id; /* 发送id放进包头,用于标记顺序 */ 
            data.head.buf_size = len; /* 记录数据长度 */ 
            if(sendto(server_socket_fd, (char*)&data, sizeof(data), 0, (struct sockaddr*)&client_addr, client_addr_length) < 0) 
            { 
              perror("Send File Failed:"); 
              break; 
            } 
            /* 接收确认消息 */ 
            recvfrom(server_socket_fd, (char*)&pack_info, sizeof(pack_info), 0, (struct sockaddr*)&client_addr, &client_addr_length); 
            receive_id = pack_info.id;  
          } 
          else 
          { 
            break; 
          } 
        } 
        else 
        { 
          /* 如果接收的id和发送的id不相同,重新发送 */ 
          if(sendto(server_socket_fd, (char*)&data, sizeof(data), 0, (struct sockaddr*)&client_addr, client_addr_length) < 0) 
          { 
            perror("Send File Failed:"); 
            break; 
          } 
          /* 接收确认消息 */ 
          recvfrom(server_socket_fd, (char*)&pack_info, sizeof(pack_info), 0, (struct sockaddr*)&client_addr, &client_addr_length); 
          receive_id = pack_info.id;  
        } 
      } 
      /* 关闭文件 */ 
      fclose(fp); 
      printf("File:%s Transfer Successful!n", file_name); 
    } 
  } 
  close(server_socket_fd); 
  return 0; 
} 

client端代码如下:

/************************************************************************* 
  > File Name: client.c 
  > Author: SongLee 
 ************************************************************************/ 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
 
#define SERVER_PORT 8000 
#define BUFFER_SIZE 1024 
#define FILE_NAME_MAX_SIZE 512 
 
/* 包头 */ 
typedef struct  
{ 
  int id; 
  int buf_size; 
}PackInfo; 
 
/* 接收包 */ 
struct RecvPack 
{ 
  PackInfo head; 
  char buf[BUFFER_SIZE]; 
} data; 
 
 
int main() 
{ 
  int id = 1; 
 
  /* 服务端地址 */ 
  struct sockaddr_in server_addr; 
  bzero(&server_addr, sizeof(server_addr)); 
  server_addr.sin_family = AF_INET; 
  server_addr.sin_addr.s_addr = inet_addr("127.0.0.1"); 
  server_addr.sin_port = htons(SERVER_PORT); 
  socklen_t server_addr_length = sizeof(server_addr); 
 
  /* 创建socket */ 
  int client_socket_fd = socket(AF_INET, SOCK_DGRAM, 0); 
  if(client_socket_fd < 0) 
  { 
    perror("Create Socket Failed:"); 
    exit(1); 
  } 
 
  /* 输入文件名到缓冲区 */ 
  char file_name[FILE_NAME_MAX_SIZE+1]; 
  bzero(file_name, FILE_NAME_MAX_SIZE+1); 
  printf("Please Input File Name On Server: "); 
  scanf("%s", file_name); 
 
  char buffer[BUFFER_SIZE]; 
  bzero(buffer, BUFFER_SIZE); 
  strncpy(buffer, file_name, strlen(file_name)>BUFFER_SIZE?BUFFER_SIZE:strlen(file_name)); 
 
  /* 发送文件名 */ 
  if(sendto(client_socket_fd, buffer, BUFFER_SIZE,0,(struct sockaddr*)&server_addr,server_addr_length) < 0) 
  { 
    perror("Send File Name Failed:"); 
    exit(1); 
  } 
 
  /* 打开文件,准备写入 */ 
  FILE *fp = fopen(file_name, "w"); 
  if(NULL == fp) 
  { 
    printf("File:t%s Can Not Open To Writen", file_name);  
    exit(1); 
  } 
 
  /* 从服务器接收数据,并写入文件 */ 
  int len = 0; 
  while(1) 
  { 
    PackInfo pack_info; 
 
    if((len = recvfrom(client_socket_fd, (char*)&data, sizeof(data), 0, (struct sockaddr*)&server_addr,&server_addr_length)) > 0) 
    { 
      if(data.head.id == id) 
      { 
        pack_info.id = data.head.id; 
        pack_info.buf_size = data.head.buf_size; 
        ++id; 
        /* 发送数据包确认信息 */ 
        if(sendto(client_socket_fd, (char*)&pack_info, sizeof(pack_info), 0, (struct sockaddr*)&server_addr, server_addr_length) < 0) 
        { 
          printf("Send confirm information failed!"); 
        } 
        /* 写入文件 */ 
        if(fwrite(data.buf, sizeof(char), data.head.buf_size, fp) < data.head.buf_size) 
        { 
          printf("File:t%s Write Failedn", file_name); 
          break; 
        } 
      } 
      else if(data.head.id < id) /* 如果是重发的包 */ 
      { 
        pack_info.id = data.head.id; 
        pack_info.buf_size = data.head.buf_size; 
        /* 重发数据包确认信息 */ 
        if(sendto(client_socket_fd, (char*)&pack_info, sizeof(pack_info), 0, (struct sockaddr*)&server_addr, server_addr_length) < 0) 
        { 
          printf("Send confirm information failed!"); 
        } 
      } 
      else 
      { 
 
      } 
    } 
    else 
    { 
      break; 
    } 
  } 
 
  printf("Receive File:t%s From Server IP Successful!n", file_name); 
  fclose(fp); 
  close(client_socket_fd); 
  return 0; 
}

感兴趣的朋友可以动手测试一下该程序,相信会对大家的Linux下C语言网络编程带来一定的帮助。


    
 
 

您可能感兴趣的文章:

  • 《UNIX网络编程》这本书适合linux下的网络编程吗?
  • 那里有linux c 函数库和linux 网络编程的书下载??
  • 学习Linux的网络编程,请问使用哪个公司的linux最好?
  • 请问unix网络编程和linux网络有什么区别
  • 新手请教,linux网络编程。
  • Unix/Linux网络编程怎样来做一个项目?
  • linux环境进行网络编程的教材
  • linux下网络编程环境配置问题
  • 请推荐几本Linux/Unix网络编程的好书吧
  • 求介绍linux下的网络编程书
  • 求助linux下的网络编程问题
  • 请有经验的朋友推荐一下Linux/Nnix下网络编程方面的书,急需学习
  • 求助:linux/unix网络编程
  • 帮推荐一本linux 网络编程的经典书吧
  • 会Linux下网络编程,能找到什么样的工作,从事什么?
  • Unix/Linux网络编程可以用来做什么?一般从事什么样的职业?
  • 学习linux网络编程需要学习些内核知识吗?
  • 哪里可以免费下载Linux下C网络编程方面的好书?
  • linux 网络编程的问题
  • 请教LINUX下的网络编程?
  • Linux c++虚函数(virtual function)简单用法示例代码
  • linux下运行libnids那个自带的示例程序printall,有点问题
  • linux c 生成随机数srand函数和rand函数介绍及代码示例
  • 求LINUX下RS232通信的示例代码
  • Linux系统命令:find(文件查找命令)介绍及用法示例
  • linux使用shell脚本,如何创建用户,并设置用户密码?能否给出示例?
  • 诚求<<LINUX编程宝典>>的示例代码!!!
  • unp(unix network programming)中的示例是针对 BSD 的,如何移植到 LINUX 上来?
  • 谁有《linux设备驱动程序》书的示例代码,发给我好么?
  • linux c 获得当前进程的进程名和执行路径(示例)
  • linux增加iptables防火墙规则的示例
  •  
    本站(WWW.)旨在分享和传播互联网科技相关的资讯和技术,将尽最大努力为读者提供更好的信息聚合和浏览方式。
    本站(WWW.)站内文章除注明原创外,均为转载、整理或搜集自网络。欢迎任何形式的转载,转载请注明出处。












  • 相关文章推荐
  • 纯C/C++ 有没有文件传输的SAMPLE(linux)
  • 请问 linux与windows两个平台下如何实现高速数据传输?
  • LINUX下如何查看正在传输的文件??
  • 在linux怎么编写文件传输
  • 用linux(服)windows(客)传输文件,windows端可以,linux端不可以,怎么回事???
  • 同一主机windows和linux文件传输
  • 有人做过Linux下的蓝牙传输吗,100分,先给20
  • linux无线网络传输也可以直接使用socket接口编程吗?
  • linux 下如何图片如何通过socket传输?
  • Linux 驱动事件传输?
  • 跪求linux下远程视频实时传输程序
  • linux文件传输
  • 两台linux之间如何传输文件
  • 关于linux下的文件加密传输。
  • linux下c编程串口传输图片,help!!
  • 请教各位大虾:在linux下,怎样用程序实现基于ftp传输文件?
  • LINUX下如何通过电话线传输数据
  • 请教一个完整的linux下通过串口传输各种文件的源码,多谢!
  • linux下的远程控制如何解决声音的传输?
  • redhat linux 超大文件传输老中断
  • linux c/c++ IP字符串转换成可比较大小的数字
  • 在win分区上安装linux和独立分区安装linux有什么区别?可以同时安装吗?(两个linux系统)
  • linux哪个版本好?linux操作系统版本详细介绍及选择方案推荐
  • 在虚拟机上安装的linux上,能像真的linux系统一样开发linux程序么?
  • secureCRT下Linux终端汉字乱码解决方法
  • 我重装window后,把linux的引导区覆盖了,进不了linux怎么办?急啊,望热心的人帮助 (现在有linux的盘)
  • Linux c字符串中不可打印字符转换成16进制
  • 安装vmware软件,不用再安装linux系统,就可以模拟linux系统了,然后可以在其上学习一下LINUX下的基本操作 了?
  • Linux常用命令介绍:更改所属用户群组或档案属性
  • 红旗Linux主机可以通过127.0.0.1访问,但如何是连网的Win2000机器通过Linux的IP去访问Linux
  • linux命令大全详细分类介绍及常用linux命令文档手册下载


  • 站内导航:


    特别声明:169IT网站部分信息来自互联网,如果侵犯您的权利,请及时告知,本站将立即删除!

    ©2012-2021,,E-mail:www_#163.com(请将#改为@)

    浙ICP备11055608号-3