这个系列是准备讲基于Linux Socket进行文件传输。简单的文件传输就是客户端可以上传文件,可以从服务器端下载文件。就这么两个功能如果再加上身份验证,就成了FTP服务器了,如果对用户的操作再加上一些功能(如分享),就可以作为一个最简单的网盘了。想想是不是有点小激动啊。

  我这一小节就不讲那么高级的东西,就先了解文件怎么传输,我们以前的聊天程序传输数据都是一次发送就完成本次的发送,因为一个sendBuf是足够的。但是对于二进制文件来说,文件的大小就不一定了,有可能很大,所以我们的Buf是不知道要多少的。所以要传输大文件,就要分多次传输,然后在目的地进行合并。这样就可以实现大文件传输了。传输的方法有两种,一种是串行一种是并行。我接下来要实现的代码是使用串行传输的,因为比较简单,而并行传输,比较复杂,要传输几个控制信号,保证数据合并重整后不会出错。并行可以使用多进(线)程。优点是可以同时传输多个文件,串行是只能一次传输一个文件。我们看一下网盘,大多是可以下载多个文件(多任务)是吧?应该用的就是并行了。如果只是从原理上讲串行的传输速度肯定是比并行的快,因为并行还要控制信号要传,而且还要重新合并整合成一个文件。既然这样那为什么所有的网盘都是使用多任务下载呢?(多个任务同时下载,每个任务同时还有多个端口在接收数据)。是因为我们所处的网络环境问题。我们的网络环境并不是那么的稳定,如果只是用一个端口进行串行传输,那么如果网络突然中断或者什么原因,导致服务器与客户端暂时连接断开。那么我们之前传的数据就要重传了。这就是为什么今天下载个任务到80%,然后明天还可以接着下载了。如果是一次性串行传输就不行了。你就会问,那在串行传输再加个控制信号不就行了?那这样的话,与并行区别就不大了。

  废话说了一大堆,到了真正要写代码的时候我还是使用串行来写,因为比较容易实现。(求原谅!)

  实现客户端向服务器发送一个指定的二进制文件

  client.cpp (为什么用cpp了,因为上次用c,然后代码越写越多,结果很多变量都写在开头,找起来不是很方便,所以用cpp了,哎,用c++就是为了用这个,会不会被骂?)

  1. #include <netinet/in.h> // sockaddr_in
  2. #include <sys/types.h> //socket
  3. #include <sys/socket.h> //socket
  4. #include <netdb.h> //gethostbyname
  5. #include <unistd.h> //close
  6. #include <stdio.h>
  7. #include <stdlib.h>
  8. #include <string.h>
  9. #include <time.h>
  10. #include <arpa/inet.h> //inet_addr
  11.  
  12. #define SERVVER_PORT 12138
  13. #define LISTEN_QUEUE 20
  14. #define BUFFER_SIZE 1024
  15.  
  16. struct Addr
  17. {
  18. char host[];
  19. int port;
  20. };
  21.  
  22. int file_push(struct Addr addr,char *filenames)
  23. {
  24. struct sockaddr_in servAddr;
  25. struct hostent * host;
  26. int sockfd;
  27. FILE *fp;
  28.  
  29. host=gethostbyname(addr.host);
  30. servAddr.sin_family=AF_INET;
  31. servAddr.sin_addr=*((struct in_addr *)host->h_addr);
  32. //servAddr.sin_addr.s_addr=inet_addr("127.0.0.1");
  33. servAddr.sin_port=htons(addr.port);
  34. if(host==NULL)
  35. {
  36. perror("获取IP地址失败");
  37. exit(-);
  38. }
  39. if((sockfd=socket(AF_INET,SOCK_STREAM,))==-)
  40. {
  41. perror("socket创建失败");
  42. exit(-);
  43. }
  44.  
  45. if(connect(sockfd,(struct sockaddr *)&servAddr,sizeof(struct sockaddr_in))== -)
  46. {
  47. perror("connect 失败");
  48. exit(-);
  49. }
  50.  
  51. //打开文件
  52. if((fp=fopen(filenames,"rb"))==NULL)
  53. {
  54. perror("文件打开失败");
  55. exit(-);
  56. }
  57. char buffer[BUFFER_SIZE];
  58. bzero(buffer,BUFFER_SIZE);
  59. printf("正在传输文件");
  60. int len=;
  61. //不断的读取文件直到文件结束
  62. while((len=fread(buffer,,BUFFER_SIZE,fp))>)
  63. {
  64. if(send(sockfd,buffer,len,)<)
  65. {
  66. perror("发送数据失败");
  67. exit(-);
  68. }
  69. bzero(buffer,BUFFER_SIZE);
  70. printf(".");//1K打印一个点//如果要实现百分比,就要计算文件大小,然后再处理即可
  71. }
  72.  
  73. fclose(fp);//关闭文件流
  74. close(sockfd);//关闭socket连接
  75.  
  76. return ;
  77. }
  78.  
  79. int main(int argc,char *argv[])
  80. {
  81. char orderbuf[BUFFER_SIZE];
  82. char orderch[BUFFER_SIZE];
  83. struct Addr addr;
  84.  
  85. strcpy(addr.host,argv[]);
  86. addr.port=atoi(argv[]);
  87. while()
  88. {
  89. printf("\n请输入文件名:");
  90. fgets(orderbuf,BUFFER_SIZE,stdin);
  91. orderbuf[strlen(orderbuf)-]=;//去掉获取到的回车符
  92. //printf("%s\n",orderbuf);
  93. file_push(addr,orderbuf);
  94. }
  95.  
  96. return ;
  97. }

  server.cpp

  1. #include <netinet/in.h>
  2. #include <sys/types.h>
  3. #include <sys/socket.h>
  4. #include <stdlib.h>
  5. #include <time.h>
  6. #include <string.h>
  7. #include <unistd.h>
  8. #include <stdio.h>
  9. #include <arpa/inet.h> //inet_ntoa
  10.  
  11. #define SERVER_PORT 12138
  12. #define LISTEN_QUEUE 20
  13. #define BUFFER_SIZE 1024
  14.  
  15. void print_time(char *ch);//打印时间
  16.  
  17. int main(int argc,char *argv[])
  18. {
  19. struct sockaddr_in server_addr;
  20. bzero(&server_addr,sizeof(server_addr));
  21. server_addr.sin_family=AF_INET;
  22. server_addr.sin_addr.s_addr=htons(INADDR_ANY);
  23. server_addr.sin_port=htons(SERVER_PORT);
  24.  
  25. //创建套接字
  26. int sockfd=socket(AF_INET,SOCK_STREAM,);
  27. if(sockfd<)
  28. {
  29. perror("创建套接字失败");
  30. exit(-);
  31. }
  32.  
  33. if(bind(sockfd,(struct sockaddr *)&server_addr,sizeof(server_addr))==-)
  34. {
  35. perror("bind 失败");
  36. exit(-);
  37. }
  38.  
  39. if(listen(sockfd,LISTEN_QUEUE))
  40. {
  41. perror("listen 失败");
  42. exit(-);
  43. }
  44.  
  45. while()
  46. {
  47. pid_t pid;
  48. struct sockaddr_in client_addr;
  49. socklen_t length=sizeof(client_addr);
  50. int clientfd=accept(sockfd,(struct sockaddr *)&client_addr,&length);
  51. if(clientfd==-)
  52. {
  53. perror("accept 失败");
  54. continue;
  55. }
  56. else
  57. {
  58. printf("客户端%s:%d连接成功\n",inet_ntoa(client_addr.sin_addr),ntohs(client_addr.sin_port));
  59. pid=fork();
  60. if(pid<)
  61. {
  62. perror("创建进程失败");
  63. }
  64. else if(pid==)/*child*/
  65. {
  66. char buffer[BUFFER_SIZE];
  67. int data_len;
  68. FILE * fp=NULL;
  69. bzero(buffer,BUFFER_SIZE);
  70. if((fp=fopen("data","wb"))==NULL)
  71. {
  72. perror("文件打开失败");
  73. exit(-);
  74. }
  75. //循环接收数据
  76. int size=;//表示有多少个块
  77. while(data_len=recv(clientfd,buffer,BUFFER_SIZE,))//data_len为0时结束,是因为当客户端没有再发送数据过来时是接收0的,也表示该文件传输完毕了。
  78. {
  79. if(data_len<)
  80. {
  81. perror("接收数据错误");
  82. exit(-);
  83. }
  84. size++;
  85. if(size==)
  86. {
  87. printf("正在接收来自%s:%d的文件\n",inet_ntoa(client_addr.sin_addr),ntohs(client_addr.sin_port));
  88. }
  89. else
  90. {
  91. printf(".");
  92. }
  93. //向文件中写入
  94. int write_len=fwrite(buffer,sizeof(char),data_len,fp);//向文件中写入,默认文件打开时会有一个文件指针进行写入。如果是并行传输就要修改这个文件指针了,还是有点麻烦的,如果是串行的,我们都不用管了,多方便。
  95. if(write_len>data_len)
  96. {
  97. perror("写入数据错误");
  98. exit(-);
  99. }
  100. bzero(buffer,BUFFER_SIZE);
  101. }
  102. if(size>)
  103. {
  104. printf("\n%s:%d的文件传送完毕\n",inet_ntoa(client_addr.sin_addr),ntohs(client_addr.sin_port));
  105. }
  106. else
  107. {
  108. printf("\n%s:%d的文件传送失败\n",inet_ntoa(client_addr.sin_addr),ntohs(client_addr.sin_port));//如果传过来的文件大小为0,也是会出现这个错误
  109. }
  110. fclose(fp);
  111. //rename("data","asdf");//这里可以修改文件的名字。保存到服务器的话可以随便改个名字,防止文件名重复
  112. exit();
  113. }
  114. else /*pather*/
  115. {
  116. ;
  117. }
  118. }
  119. close(clientfd);
  120. }
  121.  
  122. return ;
  123. }
  124.  
  125. void print_time(char *ch)
  126. {
  127. time_t now;
  128. struct tm * stm;
  129. time(&now);
  130. stm=localtime(&now);
  131. sprintf(ch,"%02d:%02d:%02d\n",stm->tm_hour,stm->tm_min,stm->tm_sec);
  132. return ;
  133. }

  两部分的代码都不难理解,关键的代码就是那个while循环了。这个就不给截图了,运行是没有问题。

  这篇杜鑫先生的回答很不错,可以看一下: http://www.zhihu.com/question/21591490

  本文地址: http://www.cnblogs.com/wunaozai/p/3886588.html

Socket网络编程--小小网盘程序(1)的更多相关文章

  1. Socket网络编程--小小网盘程序(5)

    各位好呀!这一小节应该就是这个小小网盘程序的最后一小节了,这一节将实现最后的三个功能,即列出用户在服务器中的文件列表,还有删除用户在服务器中的文件,最后的可以共享文件给好友. 列出用户在服务器中的文件 ...

  2. Socket网络编程--小小网盘程序(4)

    在这一小节中实现了文件的下载,具体的思路是根据用户的uid和用户提供的文件名filename联合两张表,取得md5唯一标识符,然后操作这个标识符对应的文件发送给客户端. 实现下载的小小网盘程序 cli ...

  3. Socket网络编程--小小网盘程序(2)

    这一节将不会介绍太多的技术的问题,这节主要是搭建一个小小的框架,为了方便接下来的继续编写扩展程序.本次会在上一小节的基础上加上一个身份验证的功能. 因为网盘程序不像聊天程序,网盘是属于主动向服务器拉取 ...

  4. Socket网络编程--小小网盘程序(3)

    接上一小节,这次增加另外的两张表,用于记录用户是保存那些文件.增加传上来的文件的文件指纹,使用MD5表示. 两张表如下定义: create table files( fid int, filename ...

  5. linux下C语言socket网络编程简例

    原创文章,转载请注明转载字样和出处,谢谢! 这里给出在linux下的简单socket网络编程的实例,使用tcp协议进行通信,服务端进行监听,在收到client的连接后,发送数据给client:clie ...

  6. 5.3linux下C语言socket网络编程简例

    原创文章,转载请注明转载字样和出处,谢谢! 这里给出在Linux下的简单socket网络编程的实例,使用tcp协议进行通信,服务端进行监听,在收到客户端的连接后,发送数据给客户端:客户端在接受到数据后 ...

  7. Socket网络编程--聊天程序(9)

    这一节应该是聊天程序的最后一节了,现在回顾我们的聊天程序,看起来还有很多功能没有实现,但是不管怎么说,都还是不错的.这一节我们将讲多服务器问题(高大上的说法就是负载问题了.)至于聊天程序的文件发送(也 ...

  8. Socket网络编程系列教程序

    C语言的用途相当多,可以用在数据结构.数据库.网络.嵌入式等方面,历经40多年不衰,真是厉害!最近一直想从某一应用方面写一个系列教程,好好地把某一方面讲深讲透.         正好博主对网络方面的编 ...

  9. Socket网络编程--FTP客户端

    Socket网络编程--FTP客户端(1)(Windows) 已经好久没有写过博客进行分享了.具体原因,在以后说. 这几天在了解FTP协议,准备任务是写一个FTP客户端程序.直接上干货了. 0.了解F ...

随机推荐

  1. eric6中ui文件编译失败,提示找不到puicc5

    1解决办法 在setting中——preference 找到qt设置——pyQT工具文件选择更改为: 我的pyuicc5.exe文件在这个目录下 然后右击编译窗口,就成功了. 如果找不到ui文件,在窗 ...

  2. Mac配置Eclipse CDT的Debug出现的问题(转)

      问题1:出现 Could not determine GDB version using command: gdb --version 原因: mac上没有安装gdb或者gdb位置配置有问题 解决 ...

  3. python 常用模块之random,os,sys 模块

    python 常用模块random,os,sys 模块 python全栈开发OS模块,Random模块,sys模块 OS模块 os模块是与操作系统交互的一个接口,常见的函数以及用法见一下代码: #OS ...

  4. Ubuntu urllib2.URLError:<urlopen error unknown url type:https>

    描述: python中urllib2 下载网页时,出现错误urllib2.URLError:<urlopen error unknown url type:https> 解决方法: pyt ...

  5. 获取AFP服务信息

    获取AFP服务信息   如果苹果系统开放TCP 548端口,说明其开启了AFP服务.这个时候,可以使用Nmap的afp-serverinfo脚本获取对应的服务信息.获取的信息包括服务名.机器类型.AF ...

  6. collections集合模块 [namedtuple,deque,*]

    collections是Python内建的一个集合模块,提供了许多有用的集合类. namedtuple namedtuple是一个函数, 它用来创建一个自定义的tuple对象,并且规定了 tuple元 ...

  7. BZOJ.3238.[AHOI2013]差异(后缀自动机 树形DP/后缀数组 单调栈)

    题目链接 \(Description\) \(Solution\) len(Ti)+len(Tj)可以直接算出来,每个小于n的长度会被计算n-1次. \[\sum_{i=1}^n\sum_{j=i+1 ...

  8. Struts2标签里面调用java方法

    <s:if test="#session.user.hasPrivilegeByName(name)"> hasPrivilegeByName(name) 为User类 ...

  9. Filebeat 快速开始

    Filebeat可以做什么 条目 filebeat 编写语言 GO 是否支持多输出 支持 是否支持多输入 支持 是否支持修改日志内容 支持 是否会丢数据 不会 对多行文件的合并 支持 对多层目录的模糊 ...

  10. STL——模拟实现空间配置器

    目录 问题 SGI版本空间配置器-std::alloc 一级空间配置器 二级空间配置器 Refill.chunkAlloc函数 最后,配置器封装的simple_alloc接口 问题 我们在日常编写C+ ...