手机端网站思路,网站开发弹窗制作,html5移动端手机网站开发流程图,怎么在微信做企业网站http://www.cnblogs.com/wunaozai/p/3886588.html 这个系列是准备讲基于Linux Socket进行文件传输。简单的文件传输就是客户端可以上传文件#xff0c;可以从服务器端下载文件。就这么两个功能如果再加上身份验证#xff0c;就成了FTP服务器了#xff0c;如果对用户的操作再…http://www.cnblogs.com/wunaozai/p/3886588.html 这个系列是准备讲基于Linux Socket进行文件传输。简单的文件传输就是客户端可以上传文件可以从服务器端下载文件。就这么两个功能如果再加上身份验证就成了FTP服务器了如果对用户的操作再加上一些功能(如分享)就可以作为一个最简单的网盘了。想想是不是有点小激动啊。 我这一小节就不讲那么高级的东西就先了解文件怎么传输我们以前的聊天程序传输数据都是一次发送就完成本次的发送因为一个sendBuf是足够的。但是对于二进制文件来说文件的大小就不一定了有可能很大所以我们的Buf是不知道要多少的。所以要传输大文件就要分多次传输然后在目的地进行合并。这样就可以实现大文件传输了。传输的方法有两种一种是串行一种是并行。我接下来要实现的代码是使用串行传输的因为比较简单而并行传输比较复杂要传输几个控制信号保证数据合并重整后不会出错。并行可以使用多进(线)程。优点是可以同时传输多个文件串行是只能一次传输一个文件。我们看一下网盘大多是可以下载多个文件(多任务)是吧应该用的就是并行了。如果只是从原理上讲串行的传输速度肯定是比并行的快因为并行还要控制信号要传而且还要重新合并整合成一个文件。既然这样那为什么所有的网盘都是使用多任务下载呢(多个任务同时下载每个任务同时还有多个端口在接收数据)。是因为我们所处的网络环境问题。我们的网络环境并不是那么的稳定如果只是用一个端口进行串行传输那么如果网络突然中断或者什么原因导致服务器与客户端暂时连接断开。那么我们之前传的数据就要重传了。这就是为什么今天下载个任务到80%然后明天还可以接着下载了。如果是一次性串行传输就不行了。你就会问那在串行传输再加个控制信号不就行了那这样的话与并行区别就不大了。 废话说了一大堆到了真正要写代码的时候我还是使用串行来写因为比较容易实现。(求原谅!) 实现客户端向服务器发送一个指定的二进制文件 client.cpp (为什么用cpp了因为上次用c然后代码越写越多结果很多变量都写在开头找起来不是很方便所以用cpp了哎用c就是为了用这个会不会被骂) 1 #include netinet/in.h // sockaddr_in2 #include sys/types.h //socket3 #include sys/socket.h //socket4 #include netdb.h //gethostbyname5 #include unistd.h //close6 #include stdio.h7 #include stdlib.h8 #include string.h9 #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[64];
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 hostgethostbyname(addr.host);
30 servAddr.sin_familyAF_INET;
31 servAddr.sin_addr*((struct in_addr *)host-h_addr);
32 //servAddr.sin_addr.s_addrinet_addr(127.0.0.1);
33 servAddr.sin_porthtons(addr.port);
34 if(hostNULL)
35 {
36 perror(获取IP地址失败);
37 exit(-1);
38 }
39 if((sockfdsocket(AF_INET,SOCK_STREAM,0))-1)
40 {
41 perror(socket创建失败);
42 exit(-1);
43 }
44
45 if(connect(sockfd,(struct sockaddr *)servAddr,sizeof(struct sockaddr_in)) -1)
46 {
47 perror(connect 失败);
48 exit(-1);
49 }
50
51 //打开文件
52 if((fpfopen(filenames,rb))NULL)
53 {
54 perror(文件打开失败);
55 exit(-1);
56 }
57 char buffer[BUFFER_SIZE];
58 bzero(buffer,BUFFER_SIZE);
59 printf(正在传输文件);
60 int len0;
61 //不断的读取文件直到文件结束
62 while((lenfread(buffer,1,BUFFER_SIZE,fp))0)
63 {
64 if(send(sockfd,buffer,len,0)0)
65 {
66 perror(发送数据失败);
67 exit(-1);
68 }
69 bzero(buffer,BUFFER_SIZE);
70 printf(.);//1K打印一个点//如果要实现百分比就要计算文件大小然后再处理即可
71 }
72
73 fclose(fp);//关闭文件流
74 close(sockfd);//关闭socket连接
75
76 return 0;
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[1]);
86 addr.portatoi(argv[2]);
87 while(1)
88 {
89 printf(\n请输入文件名:);
90 fgets(orderbuf,BUFFER_SIZE,stdin);
91 orderbuf[strlen(orderbuf)-1]0;//去掉获取到的回车符
92 //printf(%s\n,orderbuf);
93 file_push(addr,orderbuf);
94 }
95
96 return 0;
97 } server.cpp 1 #include netinet/in.h2 #include sys/types.h3 #include sys/socket.h4 #include stdlib.h5 #include time.h6 #include string.h7 #include unistd.h8 #include stdio.h9 #include arpa/inet.h //inet_ntoa10 11 #define SERVER_PORT 1213812 #define LISTEN_QUEUE 2013 #define BUFFER_SIZE 102414 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_familyAF_INET;22 server_addr.sin_addr.s_addrhtons(INADDR_ANY);23 server_addr.sin_porthtons(SERVER_PORT);24 25 //创建套接字26 int sockfdsocket(AF_INET,SOCK_STREAM,0);27 if(sockfd0)28 {29 perror(创建套接字失败);30 exit(-1);31 }32 33 if(bind(sockfd,(struct sockaddr *)server_addr,sizeof(server_addr))-1)34 {35 perror(bind 失败);36 exit(-1);37 }38 39 if(listen(sockfd,LISTEN_QUEUE))40 {41 perror(listen 失败);42 exit(-1);43 }44 45 while(1)46 {47 pid_t pid;48 struct sockaddr_in client_addr;49 socklen_t lengthsizeof(client_addr);50 int clientfdaccept(sockfd,(struct sockaddr *)client_addr,length);51 if(clientfd-1)52 {53 perror(accept 失败);54 continue;55 }56 else57 {58 printf(客户端%s:%d连接成功\n,inet_ntoa(client_addr.sin_addr),ntohs(client_addr.sin_port));59 pidfork();60 if(pid0)61 {62 perror(创建进程失败);63 }64 else if(pid0)/*child*/65 {66 char buffer[BUFFER_SIZE];67 int data_len;68 FILE * fpNULL;69 bzero(buffer,BUFFER_SIZE);70 if((fpfopen(data,wb))NULL)71 {72 perror(文件打开失败);73 exit(-1);74 }75 //循环接收数据76 int size0;//表示有多少个块77 while(data_lenrecv(clientfd,buffer,BUFFER_SIZE,0))//data_len为0时结束是因为当客户端没有再发送数据过来时是接收0的也表示该文件传输完毕了。78 {79 if(data_len0)80 {81 perror(接收数据错误);82 exit(-1);83 }84 size;85 if(size1)86 {87 printf(正在接收来自%s:%d的文件\n,inet_ntoa(client_addr.sin_addr),ntohs(client_addr.sin_port));88 }89 else90 {91 printf(.);92 }93 //向文件中写入94 int write_lenfwrite(buffer,sizeof(char),data_len,fp);//向文件中写入默认文件打开时会有一个文件指针进行写入。如果是并行传输就要修改这个文件指针了还是有点麻烦的如果是串行的我们都不用管了多方便。95 if(write_lendata_len)96 {97 perror(写入数据错误);98 exit(-1);99 }
100 bzero(buffer,BUFFER_SIZE);
101 }
102 if(size0)
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(0);
113 }
114 else /*pather*/
115 {
116 ;
117 }
118 }
119 close(clientfd);
120 }
121
122 return 0;
123 }
124
125 void print_time(char *ch)
126 {
127 time_t now;
128 struct tm * stm;
129 time(now);
130 stmlocaltime(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