资源描述:
《linux udp 单播 组播 广播实现》由会员上传分享,免费在线阅读,更多相关内容在行业资料-天天文库。
linuxudp单播组播广播实现1、组播和广播需要在局域网内才能实现,另外得查看linux系统是否支持多播和广播:ifconfigUPBROADCASTMULTICASTMTU:1500跃点数:1说明该网卡支持2、发送多播包的主机需要设置网关,否则运行sendto()会出现"networkisunreachable",网卡可以随便设置,但是一定要设。还要添加路由240.0.0.0,即:routeadd-net224.0.0.0netmask240.0.0.0deveth0routeadddefaultgw"192.168.40.1"deveth03、出现:“setsockopt:Nosuchdevice”。的提示,说明多播IP设置出现问题,系统所需要的uint32_t格式的网络地址的开头不是1110,检验通不过。解决办法:在把地址字符串"*.*.*.*"转化为uint32_t时采用htonl(inet_network(“*.*.*.*”))或者inet_aton函数,inet_aton(GRUPO,&srv.sin_addr)另外有文章:http://unix-cd.com/unixcd12/article_5577.html11.3多播单播用于两个主机之间的端对端通信,广播用于一个主机对整个局域网上所有主机上的数据通信。单播和广播是两个极端,要么对一个主机进行通信,要么对整个局域网上的主机进行通信。实际情况下,经常需要对一组特定的主机进行通信,而不是整个局域网上的所有主机,这就是多播的用途。11.3.1多播的概念多播,也称为“组播”,将网络中同一业务类型主机进行了逻辑上的分组,进行数据收发的时候其数据仅仅在同一分组中进行,其他的主机没有加入此分组不能收发对应的数据。在广域网上广播的时候,其中的交换机和路由器只向需要获取数据的主机复制并转发数据。主机可以向路由器请求加入或退出某个组,网络中的路由器和交换机有选择地复制并传输数据,将数据仅仅传输给组内的主机。多播的这种功能,可以一次将数据发送到多个主机,又能保证不影响其他不需要(未加入组)的主机的其他通信。相对于传统的一对一的单播,多播具有如下的优点:q具有同种业务的主机加入同一数据流,共享同一通道,节省了带宽和服务器的优点,具有广播的优点而又没有广播所需要的带宽。q服务器的总带宽不受客户端带宽的限制。由于组播协议由接收者的需求来确定是否进行数据流的转发,所以服务器端的带宽是常量,与客户端的数量无关。q与单播一样,多播是允许在广域网即Internet上进行传输的,而广播仅仅在同一局域网上才能进行。组播的缺点:q多播与单播相比没有纠错机制,当发生错误的时候难以弥补,但是可以在应用层来实现此种功能。q多播的网络支持存在缺陷,需要路由器及网络协议栈的支持。多播的应用主要有网上视频、网上会议等。11.3.2广域网的多播多播的地址是特定的,D类地址用于多播。D类IP地址就是多播IP地址,即224.0.0.0至239.255.255.255之间的IP地址,并被划分为局部连接多播地址、预留多播地址和管理权限多播地址3类: q局部多播地址:在224.0.0.0~224.0.0.255之间,这是为路由协议和其他用途保留的地址,路由器并不转发属于此范围的IP包。q预留多播地址:在224.0.1.0~238.255.255.255之间,可用于全球范围(如Internet)或网络协议。q管理权限多播地址:在239.0.0.0~239.255.255.255之间,可供组织内部使用,类似于私有IP地址,不能用于Internet,可限制多播范围。11.3.3多播的编程多播的程序设计使用setsockopt()函数和getsockopt()函数来实现,组播的选项是IP层的,其选项值和含义参见11.5所示。表11.5多播相关的选项getsockopt()/setsockopt()的选项含义IP_MULTICAST_TTL设置多播组数据的TTL值IP_ADD_MEMBERSHIP在指定接口上加入组播组IP_DROP_MEMBERSHIP退出组播组IP_MULTICAST_IF获取默认接口或设置接口IP_MULTICAST_LOOP禁止组播数据回送1.选项IP_MULTICASE_TTL选项IP_MULTICAST_TTL允许设置超时TTL,范围为0~255之间的任何值,例如:unsignedcharttl=255;setsockopt(s,IPPROTO_IP,IP_MULTICAST_TTL,&ttl,sizeof(ttl));2.选项IP_MULTICAST_IF选项IP_MULTICAST_IF用于设置组播的默认默认网络接口,会从给定的网络接口发送,另一个网络接口会忽略此数据。例如:structin_addraddr;setsockopt(s,IPPROTO_IP,IP_MULTICAST_IF,&addr,sizeof(addr));参数addr是希望多播输出接口的IP地址,使用INADDR_ANY地址回送到默认接口。默认情况下,当本机发送组播数据到某个网络接口时,在IP层,数据会回送到本地的回环接口,选项IP_MULTICAST_LOOP用于控制数据是否回送到本地的回环接口。例如:unsignedcharloop;setsockopt(s,IPPROTO_IP,IP_MULTICAST_LOOP,&loop,sizeof(loop));参数loop设置为0禁止回送,设置为1允许回送。3.选项IP_ADD_MEMBERSHIP和IP_DROP_MEMBERSHIP加入或者退出一个组播组,通过选项IP_ADD_MEMBERSHIP和IP_DROP_MEMBER-SHIP,对一个结构structip_mreq类型的变量进行控制,structip_mreq原型如下:structip_mreq{structin_addrimn_multiaddr;/*加入或者退出的广播组IP地址*/structin_addrimr_interface;/*加入或者退出的网络接口IP地址*/};选项IP_ADD_MEMBERSHIP用于加入某个广播组,之后就可以向这个广播组发送数据或者从广播组接收数据。此选项的值为mreq结构,成员imn_multiaddr是需要加入的广播组IP地址,成员imr_interface是本机需要加入广播组的网络接口IP地址。例如:structip_mreqmreq;setsockopt(s,IPPROTO_IP,IP_ADD_MEMBERSHIP,&mreq,sizeof(mreq)); 使用IP_ADD_MEMBERSHIP选项每次只能加入一个网络接口的IP地址到多播组,但并不是一个多播组仅允许一个主机IP地址加入,可以多次调用IP_ADD_MEMBERSHIP选项来实现多个IP地址加入同一个广播组,或者同一个IP地址加入多个广播组。当imr_interface为INADDR_ANY时,选择的是默认组播接口。4.选项IP_DROP_MEMBERSHIP选项IP_DROP_MEMBERSHIP用于从一个广播组中退出。例如:structip_mreqmreq;setsockopt(s,IPPROTP_IP,IP_DROP_MEMBERSHIP,&mreq,sizeof(sreq));其中mreq包含了在IP_ADD_MEMBERSHIP中相同的值。5.多播程序设计的框架要进行多播的编程,需要遵从一定的编程框架,其基本顺序如图11.6所示。多播程序框架主要包含套接字初始化、设置多播超时时间、加入多播组、发送数据、接收数据以及从多播组中离开几个方面。其步骤如下:(1)建立一个socket。(2)然后设置多播的参数,例如超时时间TTL、本地回环许可LOOP等。(3)加入多播组。(4)发送和接收数据。(5)从多播组离开。11.3.4内核中的多播Linux内核中的多播是利用结构structip_mc_socklist来将多播的各个方面连接起来的,其示意图如图11.7所示。 图11.7多播的内核结构structinet_sock{...__u8mc_ttl;/*多播TTL*/...__u8...mc_loop:1;/*多播回环设置*/intmc_index;/*多播设备序号*/__be32mc_addr;/*多播地址*/structip_mc_socklist*mc_list;/*多播群数组*/...};q结构成员mc_ttl用于控制多播的TTL;q结构成员mc_loop表示是否回环有效,用于控制多播数据的本地发送;q结构成员mc_index用于表示网络设备的序号;q结构成员mc_addr用于保存多播的地址;q结构成员mc_list用于保存多播的群组。1.结构ip_mc_socklist结构成员mc_list的原型为structip_mc_socklist,定义如下:structip_mc_socklist{structip_mc_socklist*next;structip_mreqnmulti; unsignedintsfmode;/*MCAST_{INCLUDE,EXCLUDE}*/structip_sf_socklist*sflist;};q成员参数next指向链表的下一个节点。q成员参数multi表示组信息,即在哪一个本地接口上,加入到哪一个多播组。q成员参数sfmode是过滤模式,取值为MCAST_INCLUDE或MCAST_EXCLUDE,分别表示只接收sflist所列出的那些源的多播数据报,和不接收sflist所列出的那些源的多播数据报。q成员参数sflist是源列表。2.结构ip_mreqnmulti成员的原型为结构structip_mreqn,定义如下:structip_mreqn{structin_addrimr_multiaddr;/*多播组的IP地址*/structin_addrimr_address;/*本地址网络接口的IP地址*/intimr_ifindex;/*网络接口序号*/};该结构体的两个成员分别用于指定所加入的多播组的组IP地址,和所要加入组的那个本地接口的IP地址。该命令字没有源过滤的功能,它相当于实现IGMPv1的多播加入服务接口。3.结构ip_sf_socklist成员sflist的原型为结构structip_sf_socklist,定义如下:structip_sf_socklist{unsignedintsl_max;/*当前sl_addr数组的最大可容纳量*/unsignedintsl_count;/*源地址列表中源地址的数量*/__u32sl_addr[0];/*源地址列表*/};q成员参数sl_addr表示是源地址列表;q成员参数sl_count表示是源地址列表中源地址的数量;q成员参数sl_max表示是当前sl_addr数组的最大可容纳量(不确定)。4.选项IP_ADD_MEMBERSHIP选项IP_ADD_MEMBERSHIP用于把一个本地的IP地址加入到一个多播组,在内核中其处理过程如图11.8所示,在应用层调用函数setsockopt()函数的选项IP_ADD_MEMBE-RSHIP后,内核的处理过程如下,主要调用了函数ip_mc_join_group()。 图11.8选项IP_ADD_MEMBERSHIP的内核处理过程(1)将用户数据复制如内核。(2)判断广播IP地址是否合法。(3)查找IP地址对应的网络接口。(4)查找多播列表中是否已经存在多播地址。(5)将此多播地址加入列表。(6)返回处理值。5.选项IP_DROP_MEMBERSHIP选项IP_DROP_MEMBERSHIP用于把一个本地的IP地址从一个多播组中取出,在内核中其处理过程如图11.9所示,在应用层调用setsockopt()函数的选项IP_DROP_MEMBERSHIP后,内核的处理过程如下,主要调用了函数ip_mc_leave_group()。 图11.9选项IP_DROP_MEMBERSHIP的内核处理过程(1)将用户数据复制入内核。(2)查找IP地址对应的网络接口。(3)查找多播列表中是否已经存在多播地址。(4)将此多播地址从源地址中取出。(5)将此地址结构从多播列表中取出。(6)返回处理值。11.3.5一个多播例子的服务器端下面是一个多播服务器的例子。多播服务器的程序设计很简单,建立一个数据包套接字,选定多播的IP地址和端口,直接向此多播地址发送数据就可以了。多播服务器的程序设计,不需要服务器加入多播组,可以直接向某个多播组发送数据。下面的例子持续向多播IP地址"224.0.0.88"的8888端口发送数据"BROADCASTTESTDATA",每发送一次间隔5s。/**broadcast_server.c-多播服务程序*/ #defineMCAST_PORT8888;#defineMCAST_ADDR"224.0.0.88"//*一个局部连接多播地址,路由器不进行转发*/#defineMCAST_DATA"BROADCASTTESTDATA"/*多播发送的数据*#defineMCAST_INTERVAL5/*发送间隔时间*/intmain(intargc,char*argv){ints;structsockaddr_inmcast_addr;s=socket(AF_INET,SOCK_DGRAM,0);/*建立套接字*/if(s==-1){perror("socket()");return-1;}memset(&mcast_addr,0,sizeof(mcast_addr));/*初始化IP多播地址为0*/mcast_addr.sin_family=AF_INET;/*设置协议族类行为AF*/mcast_addr.sin_addr.s_addr=inet_addr(MCAST_ADDR);/*设置多播IP地址*/mcast_addr.sin_port=htons(MCAST_PORT);/*设置多播端口*//*向多播地址发送数据*/while(1){intn=sendto(s,/*套接字描述符*/MCAST_DATA,/*数据*/sizeof(MCAST_DATA),/*长度*/0,(structsockaddr*)&mcast_addr,sizeof(mcast_addr));if(n<0){perror("sendto()");return-2;}sleep(MCAST_INTERVAL);/*等待一段时间*/}return0;}11.3.6一个多播例子的客户端多播组的IP地址为224.0.0.88,端口为8888,当客户端接收到多播的数据后将打印出来。客户端只有在加入多播组后才能接受多播组的数据,因此多播客户端在接收多播组的数据之前需要先加入多播组,当接收完毕后要退出多播组。/**broadcast_client.c-多播的客户端*/#defineMCAST_PORT8888;#defineMCAST_ADDR"224.0.0.88"/*一个局部连接多播地址,路由器不进行转发*/ #defineMCAST_INTERVAL5/*发送间隔时间*/#defineBUFF_SIZE256/*接收缓冲区大小*/intmain(intargc,char*argv[]){ints;/*套接字文件描述符*/structsockaddr_inlocal_addr;/*本地地址*/interr=-1;s=socket(AF_INET,SOCK_DGRAM,0);/*建立套接字*/if(s==-1){perror("socket()");return-1;}/*初始化地址*/memset(&local_addr,0,sizeof(local_addr));local_addr.sin_family=AF_INET;local_addr.sin_addr.s_addr=htonl(INADDR_ANY);local_addr.sin_port=htons(MCAST_PORT);/*绑定socket*/err=bind(s,(structsockaddr*)&local_addr,sizeof(local_addr));if(err<0){perror("bind()");return-2;}/*设置回环许可*/intloop=1;err=setsockopt(s,IPPROTO_IP,IP_MULTICAST_LOOP,&loop,sizeof(loop));if(err<0){perror("setsockopt():IP_MULTICAST_LOOP");return-3;}structip_mreqmreq;/*加入广播组*/mreq.imr_multiaddr.s_addr=inet_addr(MCAST_ADDR);/*广播地址*/mreq.imr_interface.s_addr=htonl(INADDR_ANY);/*网络接口为默认*//*将本机加入广播组*/err=setsockopt(s,IPPROTO_IP,IP_ADD_MEMBERSHIP,&mreq,sizeof(mreq));if(err<0){perror("setsockopt():IP_ADD_MEMBERSHIP");return-4;} inttimes=0;intaddr_len=0;charbuff[BUFF_SIZE];intn=0;/*循环接收广播组的消息,5次后退出*/for(times=0;times<5;times++){addr_len=sizeof(local_addr);memset(buff,0,BUFF_SIZE);/*清空接收缓冲区*//*接收数据*/n=recvfrom(s,buff,BUFF_SIZE,0,(structsockaddr*)&local_addr,&addr_len);if(n==-1){perror("recvfrom()");}/*打印信息*/printf("Recv%dstmessagefromserver:%s ",times,buff);sleep(MCAST_INTERVAL);}/*退出广播组*/err=setsockopt(s,IPPROTO_IP,IP_DROP_MEMBERSHIP,&mreq,sizeof(mreq));close(s);return0;}11.2广播前面介绍的TCP/IP知识都是基于单播,即一对一的方式,本节介绍一对多的广播方式。广播是由一个主机发向一个网络上所有主机的操作方式。例如在一个局域网内进行广播,同一子网内的所有主机都可以收到此广播发送的数据。11.2.1广播的IP地址要使用广播,需要了解IPv4特定的广播地址。IP地址分为左边的网络ID部分以及右边的主机ID部分。广播地址所用的IP地址将表示主机ID的位全部设置为1。网卡正确配置以后,可以用下面的命令来显示所选用接口的广播地址。#ifconfigeth0eth0Linkencap:EthernetHWaddr00:A0:4B:06:F4:8Dinetaddr:192.168.0.1Bcast:192.168.0.255Mask:255.255.255.0UPBROADCASTRUNNINGPROMISCMULTICASTMTU:1500Metric:1RXpackets:1955errors:0dropped:0overruns:0frame:31TXpackets:1064errors:0dropped:0overruns:0carrier:0collisions:0txqueuelen:100Interrupt:9Baseaddress:0xe400 第二行输出信息说明eth0网络接口的广播地址为192.168.0.255。这个广播IP地址的前3个字节为网络ID,即192.168.0。这个地址的主机ID部分为255,值255是表示主机ID全为1的十进制数。广播地址255.255.255.255是一种特殊的广播地址,这种格式的广播地址是向全世界进行广播,但是却有更多的限制。一般情况下,这种广播类型不会被路由器路由,而一个更为特殊的广播地址,例如192.168.0.255也许会被路由,这取决于路由器的配置。通用的广播地址在不同的环境中的含义不同。例如,IP地址255.255.255.255,一些UNIX系统将其解释为在主机的所有网络接口上进行广播,而有的UNIX内核只会选择其中的一个接口进行广播。当一个主机有多个网卡时,这就会成为一个问题。如果必须向每个网络接口广播,程序在广播之前应执行下面的步骤。(1)确定下一个或第一个接口名字。(2)确定接口的广播地址。(3)使用这个广播地址进行广播。(4)对于系统中其余的活动网络接口重复执行步骤(1)~步骤(3)。在执行完这些步骤以后,就可以认为已经对每一个接口进行广播。11.2.2广播与单播的比较广播和单播的处理过程是不同的,单播的数据只是收发数据的特定主机进行处理,而广播的数据整个局域网都进行处理。例如在一个以太网上有3个主机,主机的配置如表11.4所示。表11.4某局域网中主机的配置情况主机ABCIP地址192.168.1.150192.168.1.151192.168.1.158MAC地址00:00:00:00:00:0100:00:00:00:00:0200:00:00:00:00:03单播的示意图如图11.3所示,主机A向主机B发送UDP数据报,发送的目的IP为192.168.1.151,端口为80,目的MAC地址为00:00:00:00:00:02。此数据经过UDP层、IP层,到达数据链路层,数据在整个以太网上传播,在此层中其他主机会判断目的MAC地址。主机C的MAC地址为00:00:00:00:00:03,与目的MAC地址00:00:00:00:00:02不匹配,数据链路层不会进行处理,直接丢弃此数据。 图11.3单播的以太网示意图主机B的MAC地址为00:00:00:00:00:02,与目的MAC地址00:00:00:00:00:02一致,此数据会经过IP层、UDP层,到达接收数据的应用程序。广播的示意图如图11.4所示,主机A向整个网络发送广播数据,发送的目的IP为192.168.1.255,端口为80,目的MAC地址为FF:FF:FF:FF:FF:FF。此数据经过UDP层、IP层,到达数据链路层,数据在整个以太网上传播,在此层中其他主机会判断目的MAC地址。由于目的MAC地址为FF:FF:FF:FF:FF:FF,主机C和主机B会忽略MAC地址的比较(当然,如果协议栈不支持广播,则仍然比较MAC地址),处理接收到的数据。主机B和主机C的处理过程一致,此数据会经过IP层、UDP层,到达接收数据的应用程序。 图11.4广播的以太网示意图11.2.3广播的示例本节中是一个服务器地址发现的代码,假设服务器为A,客户端为B。客户端在某个局域网启动的时候,不知道本局域网内是否有适合的服务器存在,它会使用广播在本局域网内发送特定协议的请求,如果有服务器响应了这种请求,则使用响应请求的IP地址进行连接,这是一种服务器/客户端自动发现的常用方法。1.广播例子简介如图11.5所示为使用广播的方法发现局域网上服务器的IP地址。服务器在局域网上侦听,当有数据到来的时候,判断数据是否有关键字IP_FOUND,当存在此关键字的时候,发送IP_FOUND_ACK到客户端。客户端判断是否有服务器的响应IP_FOUND请求,并判断响应字符串是否包含IP_FOUND_ACK来确定局域网上是否存在服务器,如果有服务器的响应,则根据recvfrom()函数的from变量可以获得服务器的IP地址。 图11.5利用广播进行服务器IP地址的发现2.广播的服务器端代码服务器的代码如下,服务器等待客户端向某个端口发送数据,如果数据的格式正确,则服务器会向客户端发送响应数据。0102#defineIP_FOUND"IP_FOUND"/*IP发现命令*/03#defineIP_FOUND_ACK"IP_FOUND_ACK"/*IP发现应答命令*/ 04voidHandleIPFound(void*arg)05{06#defineBUFFER_LEN3207intret=-1;08SOCKETsock=-1;09structsockaddr_inlocal_addr;/*本地地址*/10structsockaddr_infrom_addr;/*客户端地址*/11intfrom_len;12intcount=-1;13fd_setreadfd;14charbuff[BUFFER_LEN];15structtimevaltimeout;16timeout.tv_sec=2;/*超时时间2s*/17timeout.tv_usec=0;1819DBGPRINT("==>HandleIPFound ");2021sock=socket(AF_INET,SOCK_DGRAM,0);/*建立数据报套接字*/22if(sock<0)23{24DBGPRINT("HandleIPFound:socketiniterror ");25return;26}2728/*数据清零*/29memset((void*)&local_addr,0,sizeof(structsockaddr_in));/*清空内存内容*/30local_addr.sin_family=AF_INET;/*协议族*/31local_addr.sin_addr.s_addr=htonl(INADDR_ANY);/*本地地址*/32local_addr.sin_port=htons(MCAST_PORT);/*侦听端口*/33/*绑定*/34ret=bind(sock,(structsockaddr*)&local_addr,sizeof(local_addr));35if(ret!=0)36{37DBGPRINT("HandleIPFound:binderror ");38return;39}4041/*主处理过程*/42while(1)43{44/*文件描述符集合清零*/45FD_ZERO(&readfd); 46/*将套接字文件描述符加入读集合*/47FD_SET(sock,&readfd);48/*select侦听是否有数据到来*/49ret=selectsocket(sock+1,&readfd,NULL,NULL,&timeout);50switch(ret)51{52case-1:53/*发生错误*/54break;55case0:56/*超时*/57//超时所要执行的代码5859break;60default:61/*有数据到来*/62if(FD_ISSET(sock,&readfd))63{64/*接收数据*/65count=recvfrom(sock,buff,BUFFER_LEN,0,(structsockaddr*)&from_addr,&from_len);66DBGPRINT("Recvmsgis%s ",buff);67if(strstr(buff,IP_FOUND))/*判断是否吻合*/68{69/*将应答数据复制进去*/70memcpy(buff,IP_FOUND_ACK,strlen(IP_FOUND_ACK)+1);71/*发送给客户端*/72count=sendto(sock,buff,strlen(buff),0,(structsockaddr*)&from_addr,from_len);73}74}75}76}77PRINT("<==HandleIPFound ");7879return;80}服务器端分为如下步骤:q第16行和第17行定义了服务器等待的超时时间,为2s。q第29行将地址结构清零。q第30行定义地址协议族为AF_INET。 q第31行设置IP地址为任意本地地址。q第32行设置侦听的端口。q第34行将本地的地址绑定到一个套接字文件描述符上。q第42行开始为主处理过程,使用select函数,按照2s的超时时间侦听是否有数据到来。q第45行文件描述符集合清零。q第47行将套接字文件描述符加入读集合。q第49行select侦听是否有数据到来。q第50行查看select的返回值。q第52行select发生错误。q第55行select超时。q第60行有可读的数据到来。q第65行接收数据。q第67行查看接收到的数据是否匹配。q第70行复制响应数据。q第72行发送响应数据到客户端。3.广播的客户端代码广播的客户端函数代码如下,客户端向服务器端发送命令IP_FOUND,并等待服务器端的回复,如果有服务器回复,则向服务器发送IP_FOUND_ACK,否则发送10遍后退出。01#defineIP_FOUND"IP_FOUND"/*IP发现命令*/02#defineIP_FOUND_ACK"IP_FOUND_ACK"/*IP发现应答命令*/03#defineIFNAME"eth0"04voidIPFound(void*arg)05{06#defineBUFFER_LEN3207intret=-1;08SOCKETsock=-1;09intso_broadcast=1;10structifreqifr;11structsockaddr_inbroadcast_addr;/*本地地址*/12structsockaddr_infrom_addr;/*服务器端地址*/13intfrom_len;14intcount=-1;15fd_setreadfd;16charbuff[BUFFER_LEN];17structtimevaltimeout;18timeout.tv_sec=2;/*超时时间2s*/19timeout.tv_usec=0;202122sock=socket(AF_INET,SOCK_DGRAM,0);/*建立数据报套接字*/23if(sock<0)24{25DBGPRINT("HandleIPFound:socketiniterror ");26return; 27}28/*将需要使用的网络接口字符串名字复制到结构中*/29strcpy(ifr.ifr_name,IFNAME,strlen(IFNAME));30/*发送命令,获取网络接口的广播地址*/31if(ioctl(sock,SIOCGIFBRDADDR,&ifr)==-1)32perror("ioctlerror"),exit(1);33/*将获得的广播地址复制给变量broadcast_addr*/34memcpy(&broadcast_addr,&ifr.ifr_broadaddr,sizeof(structsockaddr_in));35broadcast_addr.sin_port=htons(MCAST_PORT);/*设置广播端口*/3637/*设置套接字文件描述符sock为可以进行广播操作*/38ret=setsockopt(sock,39SOL_SOCKET,40SO_BROADCAST,41&so_broadcast,42sizeofso_broadcast);4344/*主处理过程*/45inttimes=10;46inti=0;47for(i=0;i