多播(multicast),也被称为组播,那些理论的概念就不解释了,自己看百度百科吧:
多播:http://baike.baidu.com/view/378050.htm
组播:http://baike.baidu.com/view/492256.htm
MD,这帮坑爹的,明明一个东西非要分开说,我最开始先看的组播,最让人郁闷的是一说组播,到处都说“加入一个组”“组成员管理”什么的,可从来没人说怎么“加入一个组”,组是个什么,后来看了多播以后才明白什么意思的,下面还是说说我自己的理解吧,通俗些才好懂:
多播是相对与单播和广播来说了。
单播是一个地址发送一个数据包,另一个地址收到;
广播是一个地址发送一个数据包,局域网中所有地址都收到;
多播就基于单播与广播之间,一个地址发送一个数据包,局域网中部分地址收到,部分地址收不到。
那疑问就来了,怎么才能让一部分地址收到一部分地址收不到呢?
做法就是:
首先选一个多播地址(后面介绍,暂且就当他是个地址)和端口,数据接收方选择一个网卡(要连接到局域网的)监听这个多播地址和端口,数据发送方向这个多播地址和端口发送数据,这样监听了多播地址与端口的接收方就都可以收到发送方的数据了,没有监听的自然也就收不到了!!!
这个多播地址也不是可以随便用的,在最初分配网络地址时就预留了一段地址用作多播地址使用,其中有些已经被指定给了有些协议,有些是可以咱自由使用的,具体怎么分的看我上面的两个百度百科地址吧。
这样来说的话,组的概念就好理解了,一个多播地址和端口就代表一个组,监听了这个多播地址和端口就相当于加入了这个组,不监听了就代表退出了这个组,向这个多播地址和端口发送数据就代表向这个组发送数据(MD,这叫一个绕)。
下面是我从别人处截来的一段接收多播消息的代码,借此说明各种网卡与多播地址绑定监听是怎么回事,也没验证过,改天自己写一个吧:
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <string.h>
#include "my_inet.h"
#include <arpa/inet.h>
#define MAXBUF 256
#define PUERTO 5000
#define GRUPO "224.0.1.1"
int main(void)
{
int fd, n, r;
struct sockaddr_in srv, cli;
struct ip_mreq mreq;
char buf[MAXBUF];
memset( &srv, 0, sizeof(struct sockaddr_in) );
memset( &cli, 0, sizeof(struct sockaddr_in) );
memset( &mreq, 0, sizeof(struct ip_mreq) );
srv.sin_family = MY_AF_INET;
srv.sin_port = htons(PUERTO);
if( inet_aton(GRUPO, &srv.sin_addr ) < 0 ) {
perror("inet_aton");
return -1;
}
if( (fd = socket( MY_AF_INET, SOCK_DGRAM, MY_IPPROTO_UDP) ) < 0 ){
perror("socket");
return -1;
}
if( bind(fd, (struct sockaddr *)&srv, sizeof(srv)) < 0 ){ /* 绑定一个多播地址与端口 */
perror("bind");
return -1;
}
if (inet_aton(GRUPO, &mreq.imr_multiaddr) < 0) {
perror("inet_aton");
return -1;
}
inet_aton( "172.16.48.2", &(mreq.imr_interface) );
if( setsockopt(fd, SOL_IP, IP_ADD_MEMBERSHIP, &mreq,sizeof(mreq)) < 0 ){ /* 将一个网卡(172.16.48.2)绑定到socket的多播地址(224.0.1.1)上 */
perror("setsockopt");
return -1;
}
n = sizeof(cli);
while(1){
if( (r = recvfrom(fd, buf, MAXBUF, 0, (struct sockaddr *)&cli, (socklen_t*)&n)) < 0 ){ /* 正常的接收即可 */
perror("recvfrom");
}else{
buf[r] = 0;
fprintf(stdout, "Mensaje desde %s: %s", inet_ntoa(cli.sin_addr), buf);
}
}
}
下面这个地址是上面代码的出处,也有代码解释:
http://blog.csdn.net/laohuang1122/article/details/7102827