当前位置:网站首页>UDP group (multi)cast
UDP group (multi)cast
2022-08-05 06:57:00 【Potato Watermelon Large Sesame】
1. 概述
- Unicast is used for one-to-one communication between two hosts
- 广播用于一个主机对整个局域网上所有主机上的数据通信
单播和广播是两个极端,要么对一个主机进行通信,Either communicate with hosts across the LAN.实际情况下,经常需要对一组特定的主机进行通信,rather than all hosts on the LAN,At this time there is multicast.
- IP组播(Also called multicast or multicast),It is a method that allows one or more hosts to send packets to multiple hostsTCP/IP网路技术.
- 多播是 IPv6 数据包的 3 种基本目的地址类型之一,多播是一点对多点的通信, IPv6 没有采用 IPv4 中的组播术语,而是将广播看成是多播的一个特殊例子.
2.Features of multicast
Multicast can also be called Multicast This is also true UDP 的特性之一.组播是主机间一对多的通讯模式,是一种允许一个或多个组播源发送同一报文到多个接收者的技术.组播源将一份报文发送到特定的组播地址,组播地址不同于单播地址,它并不属于特定某个主机,而是属于一组主机.一个组播地址表示一个群组,需要接收组播报文的接收者都加入这个群组.
- 广播只能在局域网访问内使用,组播既可以在局域网中使用,也可以用于广域网.
- 在发送广播消息的时候,连接到局域网的客户端不管想不想都会接收到广播数据,组播可以控制发送端的消息能够被哪些接收端接收,更灵活和人性化.
- 广播使用的是广播地址,组播A multicast address is required.
- 广播和Multicast properties默认都是关闭的,如果使用需要通过 setsockopt () 函数进行设置.
- 组播A multicast address is required,在 IPv4 中它的范围从 224.0.0.0 到 239.255.255.255,并被划分为局部链接多播地址、预留多播地址和管理权限多播地址三类
- 224.0.0.0 ~ 224.0.0.255: 局部链接多播地址:是为路由协议和其它用途保留的地址,只能用于局域网中,路由器是不会转发的地址 224.0.0.0 不能用,是保留地址
- 224.0.1.0 ~ 224.0.1.255: 为用户可用的组播地址(临时组地址),可以用于 Internet 上的.
- 224.0.2.0 ~ 238.255.255.255: 用户可用的组播地址(临时组地址),全网范围内有效
- 239.0.0.0 ~ 239.255.255.255: 为本地管理组播地址,仅在特定的本地范围内有效
组播地址不属于任何服务器或个人,它有点类似一个微信群号,任何成员(组播源)往微信群(组播 IP)发送消息(组播数据),这个群里的成员(组播接收者)都会接收到此消息.
3.组播应用
3.1 点对多点应用
点对多点应用是指一个发送者,多个接收者的应用形式,这是最常见的多播应用形式.典型的应用包括:媒体广播、媒体推送、信息缓存、事件通知和状态监视等.
3.2 多点对点应用
多点对点应用是指多个发送者,一个接收者的应用形式.通常是双向请求响应应用,任何一端(多点或点)都有可能发起请求.典型应用包括:资源查找、数据收集、网络竞拍、信息询问等.
3.3 多点对多点应用
多点对多点应用是指多个发送者和多个接收者的应用形式.通常,每个接收者可以接收多个发送者发送的数据,同时,每个发送者可以把数据发送给多个接收者.典型应用包括:多点会议、资源同步、并行处理、协同处理、远程学习、讨论组、分布式交互模拟(DIS)、多人游戏等
4. Set multicast properties
If multicast is used for data transmission,Either the sender or the receiver of the message,All need to set the relevant properties,The setup function uses the same one,即:setsockopt(),函数原型如下:
int setsockopt(int sockfd, int level, int optname, const void *optval, socklen_t optlen);
4.1 发送端
The end that sends the multicast message needs to set the multicast attribute,具体的设置方式如下:
/*
struct in_addr
{
in_addr_t s_addr; // unsigned int
};
*/
/*
函数:setsockopt
描述:配置发送端UDPMulticast properties
参数:
sockfd: 用于 UDP 通信的套接字
level: 套接字级别,Setting multicast properties requires specifying this parameter as :IPPTOTO_IP
optname: Socket option name,Setting multicast properties requires specifying this parameter as :IP_MULTICAST_IF
optval: Set multicast properties,This pointer needs to point to one struct in_addr{} The struct address of the type,This structure address is used to store the multicast address,and multicast IP Addresses are stored in big endian.
optlen:optval 指针指向的内存大小,即:sizeof(struct in_addr)
返回值:函数调用成功返回 0,调用失败返回 - 1
*/
struct in_addr opt;
// The multicast address can be initialized into this structure member
inet_pton(AF_INET, GROUP_IP, &opt.s_addr);
setsockopt(fd, IPPROTO_IP, IP_MULTICAST_IF, &opt, sizeof(opt));
4.2 接收端
Because a multicast address represents a group,Therefore, all receivers who need to receive multicast packets join this group,It is the same reason that if you want to receive group messages, you must first join the group.The way to join this multicast group is as follows:
/*
struct in_addr
{
in_addr_t s_addr; // unsigned int
};
struct ip_mreqn
{
struct in_addr imr_multiaddr; // 组播地址/多播地址
struct in_addr imr_address; // 本地地址
int imr_ifindex; // The number of the network card, Each network card has a number
};
// The number of the network card must be obtained through the name of the network card: 可以通过 ifconfig 命令查看网卡名字
#include <net/if.h>
// Convert the NIC name to the NIC number, The parameter is the name of the network card, 比如: "ens33"
// The return value is the number of the network card
unsigned int if_nametoindex(const char *ifname);
*/
/*
函数:setsockopt
描述:配置接收端UDPMulticast properties
参数:
sockfd:基于 udp communication socket
level:套接字级别,To join a multicast group, this parameter needs to be specified as :IPPTOTO_IP
optname:Socket option name,To join a multicast group, this parameter needs to be specified as :IP_ADD_MEMBERSHIP
optval:加入到多播组,This pointer should point to one struct ip_mreqn{} The struct address of the type
optlen:optval 指向的内存大小,即:sizeof(struct ip_mreqn)
*/
struct ip_mreqn opt;
// Which multicast group to join, Distinguish by multicast address
inet_pton(AF_INET, GROUP_IP, &opt.imr_multiaddr.s_addr);
opt.imr_address.s_addr = htonl(INADDR_ANY);
opt.imr_ifindex = if_nametoindex("ens33");
setsockopt(fd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &opt, sizeof(opt));
5. Multicast communication process
The end that sends the multicast message needs to send the data to the multicast address and fixed port,Terminals that want to receive multicast messages need to bind the corresponding fixed ports and then join the multicast group,Finally, data sharing can be achieved.
6.代码举例
6.1 发送端代码
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <arpa/inet.h>
#define GROUP_IP "224.0.1.0"
//#define GROUP_IP "239.0.1.10"
int main()
{
// 1. 创建通信的套接字
int fd = socket(AF_INET, SOCK_DGRAM, 0);
if(fd == -1)
{
perror("socket");
exit(0);
}
// 2. Set multicast properties (After testing, it can be sent normally without setting the multicast attribute of the sender)
struct in_addr opt;
// The multicast address can be initialized into this structure member
inet_pton(AF_INET, GROUP_IP, &opt.s_addr);
setsockopt(fd, IPPROTO_IP, IP_MULTICAST_IF, &opt, sizeof(opt));
char buf[1024];
char sendaddrbuf[64];
socklen_t len = sizeof(struct sockaddr_in);
struct sockaddr_in sendaddr;
struct sockaddr_in cliaddr;
cliaddr.sin_family = AF_INET;
cliaddr.sin_port = htons(9999); // 接收端需要绑定9999端口
// 发送组播消息, A multicast address is required, It can be the same as the multicast address used to set the multicast attribute
inet_pton(AF_INET, GROUP_IP, &cliaddr.sin_addr.s_addr);
// 3. 通信
int num = 0;
while(1)
{
memset(buf, 0, sizeof(buf));
sprintf(buf, "hello, client...%d\n", num++);
// 数据广播
sendto(fd, buf, strlen(buf)+1, 0, (struct sockaddr*)&cliaddr, len);
printf("The sent multicast data: %s\n", buf);
memset(buf, 0, sizeof(buf));
recvfrom(fd, buf, sizeof(buf), 0, (struct sockaddr *)&sendaddr, &len);
printf("sendaddr:%s, prot:%d\n", inet_ntop(AF_INET, &sendaddr.sin_addr.s_addr, sendaddrbuf, sizeof(sendaddrbuf)), sendaddr.sin_port);
printf("Received multicast messages: %s\n", buf);
}
close(fd);
return 0;
}
6.2 接收端代码
#define GROUP_IP "224.0.1.0"
//#define GROUP_IP "239.0.1.10"
int main()
{
// 1. 创建通信的套接字
int fd = socket(AF_INET, SOCK_DGRAM, 0);
if(fd == -1)
{
perror("socket");
exit(0);
}
// 2. 通信的套接字和本地的IP与端口绑定
struct sockaddr_in addr;
addr.sin_family = AF_INET;
addr.sin_port = htons(9999); // 大端
addr.sin_addr.s_addr = htonl(INADDR_ANY); // 0.0.0.0
int ret = bind(fd, (struct sockaddr*)&addr, sizeof(addr));
if(ret == -1)
{
perror("bind");
exit(0);
}
// 3. 加入到多播组
#if 0 //使用struct ip_mreqn或者 struct ip_mreq Set the multicast attribute of the receiver to receive it normally
struct ip_mreqn opt;
// Which multicast group to join, Distinguish by multicast address
inet_pton(AF_INET, GROUP_IP, &opt.imr_multiaddr.s_addr);
opt.imr_address.s_addr = htonl(INADDR_ANY);
opt.imr_ifindex = if_nametoindex("ens33");
setsockopt(fd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &opt, sizeof(opt));
#else
struct ip_mreq mreq; // 多播地址结构体
mreq.imr_multiaddr.s_addr=inet_addr(GROUP_IP);
mreq.imr_interface.s_addr = htonl(INADDR_ANY);
ret=setsockopt(fd,IPPROTO_IP,IP_ADD_MEMBERSHIP,&mreq,sizeof(mreq));
#endif
char buf[1024];
char sendaddrbuf[64];
socklen_t len = sizeof(struct sockaddr_in);
struct sockaddr_in sendaddr;
// 3. 通信
while(1)
{
// 接收广播消息
memset(buf, 0, sizeof(buf));
// 阻塞等待数据达到
recvfrom(fd, buf, sizeof(buf), 0, (struct sockaddr *)&sendaddr, &len);
printf("sendaddr:%s, prot:%d\n", inet_ntop(AF_INET, &sendaddr.sin_addr.s_addr, sendaddrbuf, sizeof(sendaddrbuf)), sendaddr.sin_port);
printf("Received multicast messages: %s\n", buf);
sendto(fd, buf, strlen(buf)+1, 0, (struct sockaddr *)&sendaddr, len);
}
close(fd);
return 0;
}
原文链接:https://blog.csdn.net/weixin_44522306/article/details/119680394
边栏推荐
猜你喜欢
随机推荐
数据库多表关联插入数据
DevExpress中针对指定列进行百分比转换
lingo入门——河北省第三届研究生建模竞赛B题
【考研结束第一天,过于空虚,想对自己进行总结一下】
Email management Filter emails
Source code analysis of Nacos configuration service (full)
【网友真实投稿】为女友放弃国企舒适圈,转行软件测试12k*13薪
1、Citrix XenDesktop 2203之AD域系统安装(一)
Collision, character controller, Cloth components (cloth), joints in the Unity physics engine
技术分析模式(十一)如何交易头肩形态
LaTeX image captioning text column automatic line wrapping
基于KECA-IGWO-KELM的间歇过程故障诊断方法
Shared memory + inotify mechanism to achieve multi-process low-latency data sharing
ndk编译so库
typescript62-泛型工具类型(record)
typescript63-索引签名类型
cs231n learning record
【Go】IM系统Centrifugo
(JLK105D)中山爆款LED恒流电源芯片方案
D45_Camera assembly Camera