socket之UDP组播(多播)

article/2025/9/9 21:48:06

1. 概述
1.1 单播用于两个主机间单对单的通信
1.2广播用于一个主机对整个局域网上所有主机上的数据通信
1.3单播和广播是两个极端,要么对一个主机进行通信,要么对整个局域网的主机进行通信
1.4实际情况下,经常需要对一组特定的主机进行通信,而不是所有局域网上的主机,这时候就有了组播
1.5IP组播(也称多址广播或多播),是一种允许一台或多台主机发送数据包到多台主机的TCP/IP网路技术。
1.6多播是 IPv6 数据包的 3 种基本目的地址类型之一,多播是一点对多点的通信, IPv6 没有采用 IPv4 中的组播术语,而是将广播看成是多播的一个特殊例子。

2.组播的特点
组播也可以称之为多播这也是 UDP 的特性之一。组播是主机间一对多的通讯模式,是一种允许一个或多个组播源发送同一报文到多个接收者的技术。组播源将一份报文发送到特定的组播地址,组播地址不同于单播地址,它并不属于特定某个主机,而是属于一组主机。一个组播地址表示一个群组,需要接收组播报文的接收者都加入这个群组。

广播只能在局域网访问内使用,组播既可以在局域网中使用,也可以用于广域网
在发送广播消息的时候,连接到局域网的客户端不管想不想都会接收到广播数据,组播可以控制发送端的消息能够被哪些接收端接收,更灵活和人性化。
广播使用的是广播地址,组播需要使用组播地址。
广播和组播属性默认都是关闭的,如果使用需要通过 setsockopt () 函数进行设置。
组播需要使用组播地址,在 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. 设置组播属性
如果使用组播进行数据的传输,不管是消息发送端还是接收端,都需要进行相关的属性设置,设置函数使用的是同一个,即:setsockopt(),函数原型如下:

int setsockopt(int sockfd, int level, int optname, const void *optval, socklen_t optlen);

4.1 发送端
发送组播消息的一端需要设置组播属性,具体的设置方式如下:

/*
struct in_addr
{in_addr_t s_addr;	// unsigned int
};
*//*
函数:setsockopt
描述:配置发送端UDP组播属性
参数:sockfd:	用于 UDP 通信的套接字level:		套接字级别,设置组播属性需要将该参数指定为:IPPTOTO_IPoptname: 	套接字选项名,设置组播属性需要将该参数指定为:IP_MULTICAST_IFoptval:	设置组播属性,这个指针需要指向一个 struct in_addr{} 类型的结构体地址,这个结构体地址用于存储组播地址,并且组播 IP 地址的存储方式是大端的。optlen:optval 指针指向的内存大小,即:sizeof(struct in_addr)返回值:函数调用成功返回 0,调用失败返回 - 1	
*/struct in_addr opt;// 将组播地址初始化到这个结构体成员中即可inet_pton(AF_INET, GROUP_IP, &opt.s_addr);setsockopt(fd, IPPROTO_IP, IP_MULTICAST_IF, &opt, sizeof(opt));

4.2 接收端
因为一个组播地址表示一个群组,所以需要接收组播报文的接收者都加入这个群组,和想要接收群消息就必须要先入群是一个道理。加入到这个组播群组的方式如下:

/*
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;              // 网卡的编号, 每个网卡都有一个编号
};
// 必须通过网卡名字才能得到网卡的编号: 可以通过 ifconfig 命令查看网卡名字
#include <net/if.h>
// 将网卡名转换为网卡的编号, 参数是网卡的名字, 比如: "ens33"
// 返回值就是网卡的编号
unsigned int if_nametoindex(const char *ifname);
*//*
函数:setsockopt
描述:配置接收端UDP组播属性
参数:sockfd:基于 udp 的通信的套接字level:套接字级别,加入到多播组该参数需要指定为:IPPTOTO_IPoptname:套接字选项名,加入到多播组该参数需要指定为:IP_ADD_MEMBERSHIPoptval:加入到多播组,这个指针应该指向一个 struct ip_mreqn{} 类型的结构体地址optlen:optval 指向的内存大小,即:sizeof(struct ip_mreqn)
*/struct ip_mreqn opt;// 要加入到哪个多播组, 通过组播地址来区分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. 组播通信流程
发送组播消息的一端需要将数据发送到组播地址和固定的端口上,想要接收组播消息的终端需要绑定对应的固定端口然后加入到组播的群组,最终就可以实现数据的共享。

在这里插入图片描述

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. 设置组播属性 (经测试可以不设置发送端组播属性也能正常发送)struct in_addr opt;// 将组播地址初始化到这个结构体成员中即可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端口// 发送组播消息, 需要使用组播地址, 和设置组播属性使用的组播地址一致就可以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("发送的组播的数据: %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("接收到的组播消息: %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.0int ret = bind(fd, (struct sockaddr*)&addr, sizeof(addr));if(ret == -1){perror("bind");exit(0);}// 3. 加入到多播组#if 0 //使用struct ip_mreqn或者 struct ip_mreq 设置接收端组播属性都可以正常接收struct ip_mreqn opt;// 要加入到哪个多播组, 通过组播地址来区分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));#elsestruct 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));#endifchar 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("接收到的组播消息: %s\n", buf);sendto(fd, buf, strlen(buf)+1, 0, (struct sockaddr *)&sendaddr, len);}close(fd);return 0;
}

参考于: 苏丙榅:https://subingwen.cn/linux/multicast/#4-1-%E5%8F%91%E9%80%81%E7%AB%AF

参考于:https://blog.csdn.net/weixin_43790540/article/details/104244546


http://chatgpt.dhexx.cn/article/EcJfQGRm.shtml

相关文章

IP多播技术详解

文章目录 前言IP多播技术的相关基本概念IP多播地址和多播组 在局域网上进行硬件多播IP多播地址和多播MAC地址映射关系 在因特网上进行IP多播网际组管理协议IGMP多播路由选择协议 前言 随着计算机网络的发展和个人计算机的普及&#xff0c;人们能够方便的在网络上畅游&#xff…

UDP多播

一、多播概念 1.1、多播 多播又称为&#xff1a;组播。 一个人发数据&#xff0c;只有加入到多播组的人接收数据 1.2、多播的特点 1、多播地址标示一组接口 2、多播可以用于广域网使用 3、在IPv4中&#xff0c;多播是可选的 1.3、多播地址 IPv4的D类地址是多播地址…

多播(组播)

什么是多播 单播用于两个主机之间的端对端通信&#xff0c;广播用于一个主机对整个局域网上所有主机上的数据通信。单播和广播是两个极端&#xff0c;要么对一个主机进行通信&#xff0c;要么对整个局域网上的主机进行通信。实际情况下&#xff0c;经常需要对一组特定的主机…

多播与广播

多播 多播(Multicast)方式的数据传输是基于UDP完成的。因此&#xff0c;与UDP服务器端/客户端的实现方式非常接近。区别在于&#xff0c;UDP数据传输以单一目标进行&#xff0c;而多播数据同时传递到加入(注册)特定组的大量主机。换言之&#xff0c;采用多播方式时&#xff0c…

多播(IP多播-网络层)与单播

多播&#xff08;IP多播-网络层&#xff09;与单播 多播&#xff08;IP多播-网络层&#xff09;简介多播组地址&#xff08;IP地址中的D类地址&#xff09;【多播组地址——D类地址】中一些不能随意使用的地址 IP多播的分类 在局域网上进行的硬件多播网际组管理协议IGMP和多播路…

多播的概念

一、多播概述 多播&#xff1a;数据的收发仅仅在同一组中进行 &#xff08;相当于我往一个群里发&#xff0c;只有加入这个群的人才能收到&#xff09; 多播的特点&#xff1a; ①多播地址标示一组接口。 ②多播可以用于广域网使用。 ③在IPv4&#xff0c;多播是可选的。 二…

win10——telnet 开启

1.win10默认没有开启 2.找到控制面板 3.找到Telnet客户端 4.测试 5.退出 ctrl ‘]’ 之后q

windows service2008 2R 开启telnet

1&#xff0c;什么是telnet 可以把telnet当作一种通信协议&#xff0c;对于入侵者而言telnet是一种远程登录的工具。 2&#xff0c;windows开启telnet&#xff08;默认情况下windows系统的telnet功能是关闭的&#xff09; 步骤&#xff1a; ①打开控制面板 输入winR 输入contro…

windows telnet开启

windows下telnet 服务开启 工作中经常要判断服务器某个端口是否连通&#xff0c;如需要看下测试服务器中tomcat的8080端口是否连通&#xff0c;需用到命令 &#xff1a; telnet 192.168.1.101 8080 ,如图&#xff1a; 解决方案 1、telnet在win7下默认是不开启的&#xff0c;所…

服务器系统开启telnet,Telnet怎么打开 Win7/Win8系统开启Telnet服务方法图解

Telnet是管理员常用的远程登录和管理工具&#xff0c;通过在本地电脑上运行Telnet客户端服务&#xff0c;就可以远程控制远端的Telnet服务器了。在Windows 2003/XP/Vista/Win7/Win8系统中都集成有Telnet服务。过默认情况下Telnet服务是被禁用的&#xff0c;需要使用的朋友&…

linux 打开telnet登录,linux开启telnet

Centos6启telnet 安装 [rootntp1 ~]# yum install telnet telnet-server xinetd 开启xinetd自启动和启动xinetd [rootntp1 ~]# chkconfig xinetd on [rootntp1 ~]# service xinetd restart 允许root用户登录 [rootntp1 ~]# vi /etc/pam.d/remote 修改/etc/pam.d/remote&#xf…

如何保存网页上的图片原图,大图

如何保存网页上的图片原图&#xff0c;大图 步骤1&#xff1a; 使用google浏览器&#xff0c;打开网页&#xff0c;按F12&#xff08;或者打开 ‘开发者工具’&#xff09;&#xff0c;点击source&#xff0c;并选择img过滤。可以看到下面的图片链接&#xff0c;如图&#xf…

网页保存成图片

Chrome中网页保存成图片 1、F12 2、Ctrl Shift P 3、输入full firefox网页保存成图片 1、看下附加组件中是否有网页截图&#xff0c;如有&#xff0c;则直接使用即可&#xff0c;无&#xff0c;打开(或下载)就行 2、截图 &#xff08;个人觉得firefox的截图比较灵活&#x…

图片的加载、显示和保存

目录 1.预备知识 2.代码实现 1.预备知识 imread() 加载图片imwrite() 保存图像 imread() 加载图片 有两个参数&#xff0c;第一个参数为字符串形式&#xff0c;也就是图片的路径 第二个参数是一个flag&#xff0c;指定了读取图像的方式&#xff0c; 默认的是加…

图片上传-->保存图片

显示图片的img标签 获取文件读取器 var imageReadernew FileReader(); 使用正则表达式给图片过滤 regexImageFilter /^(?:image/bmp|image/gif|image/jpg|image/jpeg|image/png)$/i; 文件读取器加载时间&#xff0c;在读取文件完成后触发 imageReader.οnlοadfunction(ev…

Python爬虫 自动爬取图片并保存

一、准备工作 用python来实现对图片网站的爬取并保存&#xff0c;以情绪图片为例&#xff0c;搜索可得到下图所示 f12打开源码 在此处可以看到这次我们要爬取的图片的基本信息是在img - scr中 二、代码实现 这次的爬取主要用了如下的第三方库 import re import time import …

保存网页图片的方法

最近ui妹子经常找我....... 当然不是的....只是想让我给他弄两张网页的图片....遂FE退化切图仔.... 话不多说&#xff0c;进入正文。有些朋友因为工作需要&#xff08;比如某鹿某祖等图片好想保存下来天天舔屏&#xff0c;还得是高清的....&#xff09;又不知道如何保存网页中…

图片管理之保存图片数据

保存图片数据 在保存数据之前我们需要先获取图片关联的sku的id 1、获取sku表id 接口分析 请求方式&#xff1a; GET /meiduo_admin/skus/simple/ # -------获取sku的id--------url(rskus/simple/$, images.ImageView.as_view({get: simple})), 请求参数&#xff1a; 通过请…

图片的上传与保存

先找到数据库的用户表点击设计添加 picture 字段与 nvarchar 数据类型 2.更新数据模型&#xff0c;先找到 S_User 表删除&#xff0c;然后再从数据库更新模型并保存 3.先在视图上写出图片上传与保存需要的页面 3.1 { title: ‘头像’, templet: customUserPicture },//头像…

yarn 下载安装

注意 尽量不要使用 npm 下载 yarn 因为可能下载的不是全局的&#xff0c;而且可能没有修改注册表 导致以后下载全局的东西的时候下载失败 如果已经安装了&#xff0c;且出现全局错误提示 可以直接使用 官网安装包 他会自动覆盖你之前的安装&#xff0c;并且擦写 注册表 下…