通過廣播可以很方便地實現發送數據包給局域網中的所有主機。但廣播同樣存在一些問題,例如,頻繁地發送廣播包造成所有主機數據鏈路層都會接收並交給上層協議處理,也容易引起局域網的網絡風暴。
當發送組播數據包時,只有加入指定多播組的主機數據鏈路層纔會處理,其他主機在數據鏈路層會直接丟掉收到的數據包。換句話說,我們可以通過組播的方式和指定的若干主機通信。
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;
}
如上圖所示,是一個服務端加入組播地址,兩個客戶端給該組播地址發送數據。
廣播應該是兩個服務端加入該組播地址,一個客戶端發送的數據,這兩個服務端都可以接收到數據。這個可以自己動手調試下。