问题描述:
采用UDP协议广播传输文件。所谓UDP协议是面向无连接的,不可靠的,工作于传输层的一种协议。这里补充讲解下ISO七层网络模型。
常见的ISO七层网络模型基本可以由下图进行概括说明:
对于TCP/IP模型则将7层重新归类为4类。
应用层:TFTP,HTTP,SNMP,FTP,SMTP,DNS,Telnet 等等
传输层:TCP,UDP
网络层:IP,ICMP,OSPF,EIGRP,IGMP
数据链路层:SLIP,CSLIP,PPP,MTU
程序解析:
程序分为Server端和Client端,Server端负责广播数据,Client就是接受文件。Server运行的时候首先广播文件名的长度,Client接受到长度后,再接受文件名,最后就是接受文件内容。最后Close文件就完成了。
server端的程序步骤:
1、建立UDP套接字2、设定套接字,包括允许发送广播数据int的SO_BROADCAST等
3、设置服务端的IP地址、端口等
4、服务端通过sendto函数传送文件名长度
5、服务端通过sendto函数传送文件名
6、服务端通过open函数打开文件
7、服务端循环发送文件内容,发送完之后,关闭文件,再关闭套接字
代码:
//服务端的代码
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <string.h>
#include <fcntl.h>
//创建结果成败输出
void isException(int r,char *s){
if(r==-1){
printf("%s Error! %m \n",s);
exit(0);
}
printf("%s Success !\n",s);
}
int main(int args,char *argv[]){
int fd;
int ffd;
int r;
int size;
char *fileName;
char buf[1024];
struct sockaddr_in addr;
int opt=1;
//建立Socket
fd=socket(AF_INET,SOCK_DGRAM,0);
isException(fd,"Server Init Socket ");
//设定广播方式
setsockopt(fd,SOL_SOCKET,SO_BROADCAST,&opt,sizeof(opt));
// sock:将要被设置或者获取选项的套接字。
// level:选项所在的协议层。
// optname:需要访问的选项名。
// optval:对于getsockopt(),指向返回选项值的缓冲。对于setsockopt(),指向包含新选项值的缓冲。
// optlen:对于getsockopt(),作为入口参数时,选项值的最大长度。作为出口参数时,选项值的实际长度。对于setsockopt(),
// 现选项的长度。
//SOL_SOCKET: 基本套接口;SO_BROADCAST 允许发送广播数据 int;ptlen(选项长度) :optval 的大小
//构建IP地址
addr.sin_family=AF_INET;
addr.sin_port=htons(atoi(argv[2]));
inet_aton(argv[1],&addr.sin_addr);
//传送文件名的长度
fileName=argv[3];
size=strlen(fileName);
r=sendto(fd,&size,sizeof(size),0,(struct sockaddr*)&addr,sizeof(addr));
isException(r,"Send FileName Length ");
int k;
for(k=5;k>0;k--){
printf("%d ...\n",k);
sleep(1);
}
//传送文件名
r=sendto(fd,fileName,size,0,(struct sockaddr*)&addr,sizeof(addr));
isException(r,"Send The FileName ");
//打开文件
ffd=open(fileName,O_RDONLY);
isException(ffd,"Open File ");
//循环发送文件内容
while(r!=0){
r=read(ffd,buf,sizeof(buf));
if(r==0){
sendto(fd,buf,0,0,(struct sockaddr*)&addr,sizeof(addr));
break;
}
sendto(fd,buf,r,0,(struct sockaddr*)&addr,sizeof(addr));
}
close(ffd);
close(fd);
return 0;
}
client端程序步骤:
1、创建套接字
2、设置套接字,包括其绑定方式
3、设置IP和端口等,并进行IP地址和套接字的绑定
4、客户端recv接受文件名长度,此后再接受文件名
5、在接受到文件名之后,创建文件
6、循环接受文件内容
7、接受完之后,再关掉文件和套接字
代码:
//客户端
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <string.h>
#include <fcntl.h>
void isException(int r,char *s){
if(r==-1){
printf("%s Error! %m \n",s);
exit(0);
}
printf("%s Success !\n",s);
}
int main(int args,char *argv[]){
int fd;
int ffd;
int r;
int size;
char fileName[256];
char buf[1024];
struct sockaddr_in addr;
int opt=1;
//创建fd
fd=socket(AF_INET,SOCK_DGRAM,0);
isException(fd,"Client Init Socket ");
//设定绑定方式
setsockopt(fd,SOL_SOCKET,SO_REUSEADDR,&opt,sizeof(size));
//SO_REUSEADDR,允许套接口和一个已在使用中的地址捆绑
//构造IP
addr.sin_family=AF_INET;
addr.sin_port=htons(atoi(argv[2]));
inet_aton(argv[1],&addr.sin_addr);
r=bind(fd,(struct sockaddr*)&addr,sizeof(addr));
isException(r,"Client Bind Server ");
//接受文件名长度
r=recv(fd,&size,sizeof(size),0);
isException(r,"Receive The FileName Length ");
printf("%d \n",size);
recv(fd,fileName,size,0);
fileName[size]=0;
printf("%s\n",fileName);
//创建文件
ffd=open(fileName,O_RDWR|O_CREAT|O_EXCL,0666);
isException(ffd,"Creat File ");
while(r!=0){
r=recv(fd,buf,sizeof(buf),0);
if(r==0){
break;
}
write(ffd,buf,r);
}
close(ffd);
close(fd);
return 0;
}
编译运行:
由于是进行广播,相当于从服务端拷贝数据到客户端,那么就不能将服务端的程序和客户端的程序放在同一目录。否则会产生Creat File Error! File exists 这样的提示。
在各自的目录下面,分别编译.c文件如下:
编译客户端程序:
编译服务端程序:
注意两者所处的目录是不一样的。
客户端:注意此时该目录下面只有两个文件。
服务端:其中的liujiepeng这个文件夹即是待广播的文件。
运行如下:
需要先在客户端发起运行,然后客户端会处于等待状态。本文采用的端口为8888。
再运行服务端,此时的端口号要和客户端的端口号一致。同时,需要在其后面跟上服务端上面,待广播的文件名。
此时的客户端会收到广播信息:
同时,我们看客户端所处的文件夹,可以看出,多出个文件,该文件的内容与server中的liujiepeng中的内容是一致的。
端口选择解释:
端口随便取一个大于=1204且不在/etc/services中出现的号码,[1024,65535]都可以,常用8888。这涉及到所谓的主机安全。一台主机的端口可以分为监听端口与随机取用的高级端口。所谓监听端口就是主机开启了哪些服务,那么这个服务会在Linux系统里启用一个端口来监听客户端的请求。例如FTP服务器,就会开放21号端口,这个端口会一直启用,直到FTP服务关闭为止。所谓随机取用的高级端口就是Linux要向某个主机请求服务时,Linux主机需要启用一个端口来对外连接,那么端口号是多少?Linux会随机取用一个未被使用且端口号大于1024的端口进行连接。 所以Server/Client之间的数据传送其实就是端口与端口之间的传送。 总共有多少端口,哪些是保留端口,端口编号是由1-65535组成,所以会有65535个端口。一般而言,只有root才可以开启1-1023一内的端口,这些端口就是特殊抟口,用于保留给系统使用。至于大于1024的端口,除了给系统随机取用作为连接需求之外,也可以用来服务的监听之用。 如果1-1023的端口的程序被入侵,那将表示入侵者拥有root的权限,是因为只有root才可以开启1-1023一内的端口.这个时候就要注意主机的安全了。