组播
- 广播方式是发送给同一网段下的所有主机,过多的广播会占用大量网络带宽,造成广播风暴,影响正常通讯。
- 组播(多播),加入到多播组中的主机才可以收发消息。
- 多播方式既可以给多个主机发送,又能避免造成过多负载。
- 每台主机要到传输层才能判断组播是否要处理。
- 组播地址:D类 IP地址:224.0.0.0~239.255.255.255
组播接收端(服务器)
- 创建报式套接字(socket)
- 加入多播组(setsockopt)
- 绑定组播IP(例如:224.1.2.3或者0.0.0.0)和端口号(bind)
- 等待接收数据(recvfrom)
#include <stdio.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <string.h>
#include <unistd.h>
#include <stdlib.h>
#define PORT 2222
#define IP "224.1.2.3"
int main(int argc, const char *argv[])
{
//1.创建报式套接字
int sfd = socket(AF_INET, SOCK_DGRAM, 0);
if(sfd < 0)
{
perror("socket");
return -1;
}
//2.加入多播组
struct ip_mreq mq;
mq.imr_multiaddr.s_addr = inet_addr(IP); //组播ip
mq.imr_interface.s_addr = inet_addr("0.0.0.0"); //本机ip
if(setsockopt(sfd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &mq, sizeof(mq))<0)
{
perror("setsockopt");
return -1;
}
printf("----加入多播组成功----\n");
//2.绑定服务器ip和端口号
struct sockaddr_in sin;
sin.sin_family = AF_INET;
sin.sin_port = htons(PORT);
sin.sin_addr.s_addr = inet_addr(IP);
if(bind(sfd, (struct sockaddr*)&sin, sizeof(sin)) < 0)
{
perror("bind");
return -1;
}
char buf[128] = "";
struct sockaddr_in cin;
socklen_t addrlen = sizeof(cin);
//3.接收
while(1)
{
bzero(buf, sizeof(buf));
if(recvfrom(sfd, buf, 128, 0, (struct sockaddr*)&cin, &addrlen) < 0)
{
perror("recvfrom");
return -1;
}
printf("[%s:%d]:%s\n", (char*)inet_ntoa(cin.sin_addr),ntohs(cin.sin_port),buf);
}
//4.关闭套接字
close(sfd);
return 0;
}
组播发送端(客户端)
- 创建报式套接字(socket)
- 绑定客户端信息,非必须(bind)
- 填充接收方信息,地址为组播地址(224.0.0.0~239.255.255.255)和端口号
- 发送数据(sendto)
#include <stdio.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <string.h>
#include <unistd.h>
#include <stdlib.h>
#define PORT 2222
#define IP "224.1.2.3"
int main(int argc, const char *argv[])
{
//1.创建报式套接字
int sfd = socket(AF_INET, SOCK_DGRAM, 0);
if(sfd < 0)
{
perror("socket");
return -1;
}
//2.填充服务器ip和端口号
struct sockaddr_in sin;
sin.sin_family = AF_INET;
sin.sin_port = htons(PORT);
sin.sin_addr.s_addr = inet_addr(IP);
char buf[128] = "hello---";
socklen_t addrlen = sizeof(sin);
//3.发送
while(1)
{
bzero(buf, sizeof(buf));
printf("请输入>>>");
fgets(buf, 128, stdin);
buf[strlen(buf)-1] = 0;
if(sendto(sfd, buf, 128, 0, (struct sockaddr*)&sin, addrlen)<0)
{
perror("sendto");
return -1;
}
}
//4.关闭套接字
close(sfd);
return 0;
}
本文链接:https://shengto.top/network_programming/multicast.html
转载时须注明出处及本声明