通过广播可以很方便地实现发送数据包给局域网中的所有主机。但广播同样存在一些问题,例如,频繁地发送广播包造成所有主机数据链路层都会接收并交给上层协议处理,也容易引起局域网的网络风暴。
当发送组播数据包时,只有加入指定多播组的主机数据链路层才会处理,其他主机在数据链路层会直接丢掉收到的数据包。换句话说,我们可以通过组播的方式和指定的若干主机通信。
D类地址又被称为组播地址。每一个组播地址代表一个多播地址。
下面是简单的UDP组播代码。
服务端:
/*multicast_recv.c*/
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include<sys/types.h>
#include <sys/socket.h>
#include<netinet/in.h>
#include <arpa/inet.h>
#define N 64
int main(int argc, const char *argv[])
{
int sockfd;
char buf[N] = "0";
struct sockaddr_in myaddr, peeraddr;
struct ip_mreq mreq;
socklen_t peerlen = sizeof(peeraddr);
if((sockfd = socket(PF_INET, SOCK_DGRAM, 0)) == -1)
{
perror("socket");
exit(-1);
}
bzero(&mreq, sizeof(mreq));
mreq.imr_multiaddr.s_addr = inet_addr(argv[1]);
mreq.imr_interface.s_addr = htonl(INADDR_ANY);
if(setsockopt(sockfd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &mreq, sizeof(mreq)) < 0)
{
perror("setsockopt");
exit(-1);
}
if((sockfd = socket(PF_INET, SOCK_DGRAM, 0)) == -1)
{
perror("socket");
exit(-1);
}
bzero(&myaddr, sizeof(myaddr));
myaddr.sin_family = PF_INET;
myaddr.sin_port = htons(atoi(argv[2]));
myaddr.sin_addr.s_addr = inet_addr(argv[1]);
if(bind(sockfd, (struct sockaddr *)&myaddr, sizeof(myaddr)) < 0)
{
perror("fail to bind\n");
exit(-1);
}
while(1)
{
recvfrom(sockfd, buf, N, 0, (struct sockaddr *)&peeraddr, &peerlen);
printf("%s : %d %s\n", inet_ntoa(peeraddr.sin_addr), ntohs(peeraddr.sin_port), buf);
}
return 0;
}
客户端:
/* multicast_send.c
*/
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#define N 64
typedef struct sockaddr SA;
int main(int argc,char* argv[])
{
int sockfd;
char buf[N]="This is a multicast package\n";
struct sockaddr_in dstaddr;
if(argc < 3)
{
printf("Usage:<%s> <ip> <port>\n",argv[0]);
return -1;
}
if((sockfd = socket(PF_INET,SOCK_DGRAM,0)) == -1)
{
perror("failed to socket");
exit(-1);
}
bzero(&dstaddr,sizeof(dstaddr));
dstaddr.sin_family = PF_INET;
dstaddr.sin_port = htons(atoi(argv[2]));
dstaddr.sin_addr.s_addr = inet_addr(argv[1]);
while(1)
{
sendto(sockfd,buf,N,0,(SA *)&dstaddr,sizeof(dstaddr));
sleep(1);
}
return 0;
}
如上图所示,是一个服务端加入组播地址,两个客户端给该组播地址发送数据。
广播应该是两个服务端加入该组播地址,一个客户端发送的数据,这两个服务端都可以接收到数据。这个可以自己动手调试下。