1. 以下载链接“http://www.boa.org/boa-0.94.13.tar.gz”为例:
  2. 断点续传实验大概步骤:
  3. =====================
  4. 1,使用gethostbyname()获取站点“www.boa.org”的IP地址
  5. 2,以“boa-0.94.13.tar.gz”为例,构建HTTP请求报文首部:
  6. char *httpreq = "GET /boa-0.94.13.tar.gz HTTP/1.1\r\n"
  7. "Range: bytes=%d-%d\r\n"
  8. "Host: %s\r\n\r\n"
  9. begin, end, host);
  10. 注意:
  11. A) 以上字符串httpreq就是发送给远程主机www.boa.orgHTTP请求报文,通过TCP发送
  12. B) Range字段是要求主机发送申请文件的部分内容,beginend分别是文件的开始和结束
  13. B.1 如果写成 "Range: bytes=0-\r\n",代表要求主机发送全文
  14. B.2 如果写成 "Range: bytes=-1000\r\n",代表要求主机发送前1000个字节
  15. B.3 如果写成 "Range: bytes=0-200\r\n",代表要求主机发送前200个字节
  16. C) 正常情况下,HTTP服务器会对这个请求报文返回206,并给出实际返回的字节数和范围:
  17. C.1 "Content-Length: 1000" 这个字段代表本次HTTP发来的数据大小为1000个字节(不含HTTP首部)
  18. C.2 "Content-Range: bytes 2000-2999/9999" 代表本次传送的数据范围是第2000-2999个字节(共1000个字节),而所请求的文件总大小是9999个字节。
  19. 3,将收到的报文的HTTP首部去掉(HTTP首部是指从开头到\r\n\r\n结尾的部分),剩下的就是下载的文件内容
  20. 4,将文件内容以非缓冲方式保存下来。
  21. 5,如果发生下载时网络断线,或者人为终止了下载进程,那么在下一次下载时先获取当前已下载部分的大小,并作为Range参数告知远端HTTP服务器,要求发送部分文件,实现断点续传,节约网络流量节约时间。
  1. #include <stdio.h>
  2. #include <unistd.h>
  3. #include <stdlib.h>
  4. #include <sys/stat.h>
  5. #include <netdb.h>
  6. #include <arpa/inet.h>
  7. #include <netinet/in.h>
  8. #include <sys/socket.h>
  9. #include <string.h>
  10. #include <strings.h>
  11. #include <sys/types.h>
  12. #include <fcntl.h>
  13. #include <assert.h>
  14. #include <stdbool.h>
  15. #include "common.h"
  16. #define RWRWRW (S_IRUSR|S_IWUSR|S_IRGRP|S_IWGRP|S_IROTH|S_IWOTH)
  17. #define SERV_PORT 80
  18. #define FILENAME "boa-0.94.13.tar.gz"
  19. typedef struct sockaddr SA;
  20. void http_request(char *buf, int size, const char *filename
  21. , const int begin, const char *host);
  22. int main(void)
  23. {
  24. int sockfd;
  25. struct sockaddr_in servaddr;
  26. struct hostent *hptr = NULL;
  27. struct in_addr **pptr = NULL;
  28. if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
  29. err_quit("socket error");
  30. }
  31. char *hostname = "www.boa.org";
  32. if ((hptr = gethostbyname(hostname)) == NULL) {
  33. err_quit("gethostbyname error for host: %s: %s",
  34. hostname, hstrerror(h_errno));
  35. }
  36. pptr = (struct in_addr**)hptr->h_addr_list;
  37. bzero(&servaddr, sizeof(servaddr));
  38. servaddr.sin_family = AF_INET;
  39. servaddr.sin_port = htons(SERV_PORT);
  40. memcpy(&servaddr.sin_addr, *pptr, sizeof(struct in_addr));
  41. if (connect(sockfd, (SA *)&servaddr, sizeof(servaddr)) < 0) {
  42. err_quit("connect error");
  43. }
  44. int fd;
  45. int file_size;
  46. struct stat statbuf;
  47. if (access(FILENAME, F_OK)) {
  48. if ((fd = open(FILENAME, O_WRONLY|O_CREAT, RWRWRW)) < 0) {
  49. err_sys("create %s failure", FILENAME);
  50. }
  51. file_size = 0;
  52. } else {
  53. if ((fd = open(FILENAME, O_APPEND|O_WRONLY, RWRWRW)) < 0) {
  54. err_sys("open %s failure", FILENAME);
  55. }
  56. if (stat(FILENAME, &statbuf) < 0) {
  57. err_sys("stat error");
  58. }
  59. file_size = statbuf.st_size;
  60. }
  61. #ifdef DEBUG
  62. printf(" host:\t%s\n", hostname);
  63. printf(" filename:\t%s\n", FILENAME);
  64. printf("start size:\t%d\n", file_size);
  65. printf("------------------------------------------------\n\n");
  66. #endif
  67. char request[MAXLINE];
  68. http_request(request, sizeof(request), FILENAME, file_size, hostname);
  69. if (write(sockfd, request, strlen(request)) != strlen(request)) {
  70. err_quit("request failure");
  71. }
  72. char recvbuf[4096];
  73. int nread;
  74. bool flage = true;
  75. char tar[] = "\r\n\r\n";
  76. char *begin = NULL;
  77. while (1) {
  78. bzero(recvbuf, sizeof(recvbuf));
  79. if ((nread = read(sockfd, recvbuf, sizeof(recvbuf))) <= 0) {
  80. if (0 == nread) {
  81. break;
  82. } else {
  83. err_sys("read error");
  84. }
  85. }
  86. if (flage) {
  87. if ((begin = strstr(recvbuf, tar)) == NULL) {
  88. continue;
  89. } else {
  90. write(fd, begin+strlen(tar), nread-(begin-recvbuf)-strlen(tar));
  91. flage = false;
  92. }
  93. }
  94. if (!flage) {
  95. if (write(fd, recvbuf, nread) != nread) {
  96. err_quit("write error");
  97. }
  98. }
  99. #ifdef DEBUG
  100. printf("******");
  101. fflush(stdout);
  102. #endif
  103. }
  104. /*
  105. if (stat(FILENAME, &statbuf) < 0) {
  106. err_quit("stat failure");
  107. } else {
  108. if (statbuf.st_size == full_size) {
  109. printf("---success!---\n");
  110. }
  111. }
  112. */
  113. printf("---success!---\n");
  114. return EXIT_SUCCESS;
  115. }
  116. void http_request(char *buf, int size, const char *filename, const int begin, const char *host)
  117. {
  118. assert(buf);
  119. assert(filename);
  120. assert(host);
  121. bzero(buf, size);
  122. snprintf(buf, size, "GET /%s "
  123. "HTTP/1.1\r\n"
  124. "Range: bytes=%d-\r\n"
  125. "Host: %s\r\n\r\n", filename, begin, host);
  126. }

linux网络编程之断点传输文件的更多相关文章

  1. Android网络编程只局域网传输文件

    Android网络编程之局域网传输文件: 首先创建一个socket管理类,该类是传输文件的核心类,主要用来发送文件和接收文件 具体代码如下: package com.jiao.filesend; im ...

  2. Java 学习笔记 网络编程 使用Socket传输文件 CS模式

    Socket的简单认识 Socket是一种面向连接的通信协议,Socket应用程序是一种C/S(Client端/Server端)结构的应用程序 Socket是两台机器间通信的端点. Socket是连接 ...

  3. Linux网络编程系列-TCP传输控制

    滑动窗口(sliding window) 滑动窗口是用于流量控制的,发送端根据接收端的处理能力发送数据,不至于造成过多的丢包. 是发送方和接收方间的协调,对方的接收窗口大小就是自己的发送窗口大小. 在 ...

  4. linux网络编程-(socket套接字编程UDP传输)

    今天我们来介绍一下在linux网络环境下使用socket套接字实现两个进程下文件的上传,下载,和退出操作! 在socket套接字编程中,我们当然可以基于TCP的传输协议来进行传输,但是在文件的传输中, ...

  5. 【linux草鞋应用编程系列】_5_ Linux网络编程

    一.网络通信简介   第一部分内容,暂时没法描述,内容实在太多,待后续专门的系列文章.   二.linux网络通信     在linux中继承了Unix下“一切皆文件”的思想, 在linux中要实现网 ...

  6. Linux网络编程入门 (转载)

    (一)Linux网络编程--网络知识介绍 Linux网络编程--网络知识介绍客户端和服务端         网络程序和普通的程序有一个最大的区别是网络程序是由两个部分组成的--客户端和服务器端. 客户 ...

  7. [转] - Linux网络编程 -- 网络知识介绍

    (一)Linux网络编程--网络知识介绍 Linux网络编程--网络知识介绍客户端和服务端         网络程序和普通的程序有一个最大的区别是网络程序是由两个部分组成的--客户端和服务器端. 客户 ...

  8. Linux C 程序 Linux网络编程(21)

    Linux网络编程网络编程必备的理论基础网络模型,地址,端口,TCP/IP协议 TCP/IP协议是目前世界上使用最广泛的网络通信协议日常中的大部分应用使用该系列协议(浏览网页,收发电子邮件,QQ聊天等 ...

  9. 【转】Linux网络编程入门

    (一)Linux网络编程--网络知识介绍 Linux网络编程--网络知识介绍客户端和服务端         网络程序和普通的程序有一个最大的区别是网络程序是由两个部分组成的--客户端和服务器端. 客户 ...

随机推荐

  1. 根据从redis缓存的数据查询出来,在从数据库中取出所有的数据,俩个数据进行比较,去掉重复,剩下库中新插入的数据,取出新数据,然后把redis中的缓存数据清空把从数据库中查出来的所有数据放到redis缓存中

    参考代码: public String getNewCenter(HttpServletRequest request,HttpServletResponse resonse){ JSONObject ...

  2. PowerDesigner如何将字段的注释显示出来

    选定一个编辑的表,右键- >Properties- >Columns- >Customize Columns and Filter(或直接用快捷键Ctrl+U)- >Comme ...

  3. P2375 [NOI2014]动物园 KMP

    好,暴力能拿$50pts\space qwq$ 暴力的思路就是一直跳$nxt[j]$,直到它的长度小于串的一半,然后开始计数,当然要接着跳$nxt[j]$ 正解:考虑没有长度要求的(不要求不重合)公共 ...

  4. 什么是SG?+SG模板

    先,定义一下 状态Position P 先手必败 N x先手必胜 操作方法: 反向转移 相同状态 不同位置 的一对 相当于无 对于ICG游戏,我们可以将游戏中每一个可能发生的局面表示为一个点.并且若存 ...

  5. Codeforces Round #364 (Div. 2) A

    Description There are n cards (n is even) in the deck. Each card has a positive integer written on i ...

  6. 【手撸一个ORM】MyOrm的使用说明

    [手撸一个ORM]第一步.约定和实体描述 [手撸一个ORM]第二步.封装实体描述和实体属性描述 [手撸一个ORM]第三步.SQL语句构造器和SqlParameter封装 [手撸一个ORM]第四步.Ex ...

  7. spring boot(10) 基础学习内容

    A Spring boot(10) 基础学习内容 B SpringBoot(16) 基础学习内容

  8. java编程如何实现多条2017-01-16 22:28:11.0这样的时间数据,转换成Date类型Mon Jan 16 22:28:11 CST 2017这样的时间数据

    不多说,直接上干货! package zhouls.bigdata.DataFeatureSelection.sim; import java.text.ParseException; import ...

  9. SpringBoot | 第三章:springboot配置详解

    基于springboot的约定优于配置的原则,在多数情况下,启动一个应用时,基本上无需做太多的配置,应用就能正常启动.但在大部分开发环境下,添加额外配置是无所避免的,比如自定义应用端口号(比较在机器比 ...

  10. MVC学习6 学习使用Code First Migrations功能 把Model的更新同步到DB中

     参考:http://www.asp.net/mvc/tutorials/mvc-4/getting-started-with-aspnet-mvc4/adding-a-new-field-to-th ...