组播详解及示例代码

news2025/1/11 14:20:42

写在前面

由于公司业务需要用到组播实现,这里就记录下学习过程。在学习组播之前,我们先来看看另外两种数据包传输方式:单播和广播

  1. 单播:简单来说就是数据一对一发送,如果需要给多个主机发送数据时,就需要将同一份源数据的多次拷贝,发送给这些主机。无疑加重了源主机以及网络带宽的压力。这种传输方式不利于批量传输数据。
  2. 广播:不需要这些信息的主机也会收到该信息,数据的安全性得不到保证,还会造成同一网段内的信息泛滥,浪费带宽。可以看到,这种传输方式不利于给特定的用户传输数据。

一、什么是组播

组播(Multicast),又称“多播”,是一种数据包传输方式。它以"尽力而为"的形式发送信息到某个目标组,这个目标组称为组播组。

源主机向多个主机发送数据时,源主机只发送一份数据,数据的目的地址是组播组地址。这样,凡是属于该组的成员,都可以接收到一份源主机发送的数据的拷贝,此组播方式下,只有真正信息需要的成员会收到信息,其他主机不会收到。

组播相较于单播和广播的优势:

  1. 相较于单播,被传递的信息只会在距信息源尽可能远的网络节点才开始被复制和分发,用户的增加不会导致信息源负载的加重以及网络资源消耗的显著增加
  2. 相较于广播,被传递的信息只会发送给需要该信息的接收者,所以不会造成网络资源的浪费,并能提高信息传输的安全性

小结:当有多台主机同时成为一个数据包的接受者时,出于对带宽和CPU负担的考虑,组播成为了一种最佳选择。

1.1 组播相关术语

  • 组播组:用组播地址标识的一组主机集合
  • 组播源:数据的发送者
  • 组播成员:加入某个组播组的主机
  • 组播路由器:运行组播协议的设备

1.2 组播如何工作

组播通过把224.0.0.0-239.255.255.255的D类地址作为目的地址,有一台源主机发出目的地址是以上范围组播地址的报文,在网络中,如果有其他主机对于这个组的报文有兴趣的,可以申请加入这个组,并可以接受这个组,而其他不是这个组的成员是无法接受到这个组的报文的。
在这里插入图片描述

1.3 组播实现原理

上面说到了组播路由器,这里我们着重看下这个组播路由器的作用。

用户根据IGMP协议发送请求报文,路由器收到IGMP报文后,会把用户加入自己的组播组,组播报文到达路由器时,根据组播组复制多份数据发给组内的所有主机。

注意:IGMP报文并不是发给路由器,它的目的地址只有目标主机,报文从用户到目标主机可能经历多个路由器,用户必须加入这些路由器的组播组,为什么呢?因为只有用户加入了这条路径上所有的路由器的组播组之后,组播源发出的数据,才能在经过层层路由是转发到正确的目标用户。

发送IGMP报文需要知道组播源的IP地址,那用户是如何知道组播源的IP地址的呢?答案是:RP(Rendezvous Point)集中点,具体来说就是,让组播源知道RP的IP地址,让用户知道RP的IP地址。

获取组播源IP地址

在这里插入图片描述

  1. 组播源通过单播的方式把组播 239.0.0.2 封装在一个单播发送给RP(src_ip: 192.168.60.213, dst_ip: 192.168.60.210)
  2. 用户D向RP发送请求加入组播239.0.0.2 的IGMP报文
  3. RP收到请求后,把组播源发送的单播数据复制一份发送给用户D
  4. 用户D收到报文后解析就能拿到组播源的IP地址192.168.60.213
组播实现

在这里插入图片描述

  1. 接收端发送IGMP报文给组播源,经过的所有路由器都会把接受端加入组播组239.0.0.2
  2. 组播源发送数据到组播组 239.0.0.2
  3. 路由器收到数据,具体发给谁由路由器的路由表决定

二、组播代码示例

2.1 server端(组播源)

MulticastServer.h

#include <iostream>
#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <arpa/inet.h>
#include <net/if.h>
#include <errno.h>
#include <string>

#define SERVER_PORT 8000
#define CLIENT_PORT 9000
#define GROUP "239.0.0.2"

using namespace std;

class MulticastServer{
public:
    MulticastServer();
    ~MulticastServer();

    bool Init();
    void SendMessage(string payloadMessage);

private:
    int m_sockfd;
    struct sockaddr_in m_serveraddr, m_clientaddr;
};

MulticastServer.cpp

#include "MulticastServer.h"

bool MulticastServer::Init()
{
    m_sockfd = socket(AF_INET, SOCK_DGRAM, 0);                /*构造用于UDP通信的套接字*/
    
    bzero(&m_serveraddr, sizeof(m_serveraddr));
    m_serveraddr.sin_family = AF_INET;                        /* IPv4 */
    m_serveraddr.sin_addr.s_addr = htonl(INADDR_ANY);         /*本地任意IP INADDR_ANY = 0 */
    m_serveraddr.sin_port = htons(SERVER_PORT);

    bind(m_sockfd, (struct sockaddr *)&m_serveraddr, sizeof(m_serveraddr));

    struct ip_mreqn group;
    inet_pton(AF_INET, GROUP, &group.imr_multiaddr);        /*设置组播组的地址*/
    inet_pton(AF_INET, "0.0.0.0", &group.imr_address);      /* 本地任意IP 自动分配有效IP*/
    group.imr_ifindex = if_nametoindex("enp5s0");             /* 给出网卡名,转换为对应编号:eth0 --> 编号         ,,  命令:ip ad */

    int ret = setsockopt(m_sockfd, IPPROTO_IP, IP_MULTICAST_IF, &group, sizeof(group));  /*获取组播权限*/
    if (ret < 0) {
        printf("Fail to disable multicast loop, err: %s",strerror(errno));
        return false;
    }else{
        printf("disable multicast loop success.\n");
    }
    // ret = setsockopt(m_sockfd, IPPROTO_IP , IP_MULTICAST_LOOP, &group, sizeof(group));


    bzero(&m_clientaddr, sizeof(m_clientaddr));                 /* 构造client 地址 IP+端口号*/
    m_clientaddr.sin_family = AF_INET;
    inet_pton(AF_INET, GROUP, &m_clientaddr.sin_addr.s_addr); /* IPv4  239.0.0.2+9000 */
    m_clientaddr.sin_port = htons(CLIENT_PORT);

    return true;
}

void MulticastServer::SendMessage(string payloadMessage)
{
    // sprintf(buf, "from 192.168.60.213 server info: multicast %d\n", i++);
    //fgets(buf, sizeof(buf), stdin);
    sendto(m_sockfd, (char*)payloadMessage.c_str(), payloadMessage.size(), 0, (struct sockaddr *)&m_clientaddr, sizeof(m_clientaddr));
}

main.c

#include "MulticastServer.h"

int main(int argc, char *argv[])
{
    MultiBroadcastServer server;
    server.init();

    int idx = 0;
    while(true)
    {
        idx++;
        std::string msg = "from 192.168.60.213 server info: multicast " + to_string(idx) + "\n";
        server.SendMessage(msg);
        sleep(1);
    }
	return 0;
}

2.2 client端(接收端)

MulticastClient.h

#include <iostream>
#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <arpa/inet.h>
#include <net/if.h>
#include <errno.h>
#include <string>

#define SERVER_PORT 8000
#define CLIENT_PORT 9000
#define GROUP "239.0.0.2"

using namespace std;

class MulticastClient{
public:
    MulticastClient();
    ~MulticastClient();

    bool Init();
    void recvMessage(char* buffer, int &len);

private:
    int m_confd;
    struct sockaddr_in m_clientaddr;
};

MulticastClient.cpp

#include "MulticastClient.h"

bool MulticastClient::Init()
{
   	struct ip_mreqn group;                                                  /*组播结构体*/
    m_confd= socket(AF_INET, SOCK_DGRAM, 0);
    bzero(&m_clientaddr, sizeof(m_clientaddr));                                   /* 初始化*/
    m_clientaddr.sin_family = AF_INET;
    inet_pton(AF_INET, "0.0.0.0" , &m_clientaddr.sin_addr.s_addr);
    m_clientaddr.sin_port = htons(CLIENT_PORT);

    bind(m_confd, (struct sockaddr *)&m_clientaddr, sizeof(m_clientaddr));

    inet_pton(AF_INET, GROUP, &group.imr_multiaddr);                        /* 设置组播组地址*/
    inet_pton(AF_INET, "0.0.0.0", &group.imr_address);                      /*使用本地任意IP添加到组播组*/
    group.imr_ifindex = if_nametoindex("enp5s0");                             /* 设置网卡名 编号 ip ad */    
    setsockopt(m_confd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &group, sizeof(group));/* 将client加入组播组*/

    return true;
}

void MulticastClient::recvMessage(char* buffer, int &len)
{
	len = recvfrom(m_confd, buffer, sizeof(buffer), 0, NULL, 0);
	std::cout << "client recv: " << std::string(buffer) <<  " , len : "<<  len << std:endl;
}

main.c

#include "MulticastClient.h"

int main(int argc, char *argv[])
{
    MulticastClient client;
    client.init();
	
    while(true)
    {
    	char buffer[BUFSIZ] = {0};
    	int len = 0;
		client.recvMessage(buffer, len);
    }
	return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/388549.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Unity入门精要02---纹理

纹理和材质不可分割 本节知识结构 实践&#xff1a;简单贴一张纹理到模型上 首先在属性处添加相关属性 Properties {_Color ("Color Tint", Color) (1, 1, 1, 1)_MainTex ("Main Tex", 2D) "white" {}//加入纹理_Specular ("Specular&q…

虚拟机VirtualBox下载与安装+虚拟机配置

1、虚拟机VirtualBox下载与安装官网地址&#xff1a;Oracle VM VirtualBox进入官网&#xff0c;点击左侧Dwonlaods点击Windows hosts&#xff0c;开始下载3.打开文件.出现报错&#xff1a;Oracle VM VirtualBox 7.0.4 needs the Microsoft Visual C 2019 Redistributable Packa…

转义字符\033(设置终端的字体显示效果)

以下内容源于网络资源的学习与整理&#xff0c;如有侵权请告知删除。 参考博客 命令行特殊显示效果\033和发声音\007_华硕他哥的博客-CSDN博客 \033格式:指定输出格式_51CTO博客_wkt格式 一、\033的说明 在关于ASCII字符的那些事儿中提到&#xff0c;ASCII字符可以使用前面…

2023年天津医科大学临床医学院专升本专业课考试通知

天津医科大学到临床医学院2023年高职升本科专业课考试通知一、考试时间、地点 天津医科大学临床医学院2023年高职升本科考试定于2023年3月25日举行&#xff0c;考试地点为天津市滨海新区大港学苑路167号天津医科大学临床医学院教学楼。 二、准考证打印开放时间 本次考试打印准考…

3.4笔试总结

感觉题目好难没有做过这种类型的填空题真的是给应届实习生做的题吗第一大题 填空题一开始题目放在pdf里,我都没看清题目想干嘛其实题目看起来复杂.如果仔细读题意还是能做出几个空,无奈我太过紧张浪费了好几分钟空档时间public class Main { public static boolean isRegularMa…

内核角度谈谈Linux进程和线程

目录前言内核对进程和线程的表示创建进程的过程创建线程的过程创建进程和线程的异同揭秘 do_fork 系统调用结论前言 昨天面试的时候&#xff0c;面试官问我了个平平淡淡的问题–>“聊聊Linux中进程和线程”; 相比大家不管是在考试还是面试中或多或少都遇到过这个问题&…

python高德地图+58租房网站平台源码

wx供重浩&#xff1a;创享日记 对话框发送&#xff1a;python地图 免费获取完整源码源文件说明文档配置教程等 在PyCharm中运行《高德地图58租房》即可进入如图1所示的高德地图网页。 具体的操作步骤如下&#xff1a; &#xff08;1&#xff09;打开地图网页后&#xff0c;在编…

xgboost:防止过拟合的方法收缩和特征列、采样

xgboost除了第2.1节中提到的正则化目标之外&#xff0c;还使用了另外两种技术来进一步防止过拟合1。 目录shrinkage[^1]Column Subsampling[^2]shrinkage2 第一种技术是由弗里德曼提出的收缩。在树提升的每一步之后&#xff0c;收缩率以η因子的比例缩放新添加的权重。与随机优…

Scrapy框架(高效爬虫)

文章目录一、环境配置二、创建项目三、scrapy数据解析四、基于终端指令的持久化存储1、基于终端指令2、基于管道3、数据同时保存至本地及数据库4、基于spider爬取某网站各页面数据5、爬取本页和详情页信息&#xff08;请求传参&#xff09;6、图片数据爬取ImagesPipeline五、中…

IP欺骗种类有哪些?

每台计算机都有一个IP地址&#xff0c;发送的任何数据都被分成许多块&#xff08;“数据包”&#xff09;&#xff0c;每个数据包单独传输&#xff0c;当这些数据包到达链的末端时&#xff0c;就会重新组装并作为一个整体呈现。此外&#xff0c;每个数据包还有其可识别信息&…

4、High-Resolution Image Synthesis with Latent Diffusion Models

简介github地址diffusion model明显的缺点是耗费大量的时间、计算资源&#xff0c;为此&#xff0c;论文将其应用于强大的预训练自编码器的潜在空间 &#xff0c;这是首次允许在复杂性降低和细节保存之间达到一个近乎最佳的点&#xff0c;极大地提高了视觉保真度。通过在模型架…

操作系统复习题

什么是线程&#xff1f; 线程&#xff08;Thread&#xff09;&#xff1a;轻量级进程&#xff0c;是操作系统进行调度的最小单位。一个线程是一个任务&#xff08;一个程序段&#xff09;的一次执行过程。线程不占有内存空间&#xff0c;它包括在进程的内存空间中。在同一个进程…

自然语言处理历史最全预训练模型(部署)汇集分享

什么是预训练模型&#xff1f;预练模型是其他人为解决类似问题而创建的且已经训练好的模型。代替从头开始建立模型来解决类似的问题&#xff0c;我们可以使用在其他问题上训练过的模型作为起点。预训练的模型在相似的应用程序中可能不是100&#xff05;准确的。本文整理了自然语…

踩坑:maven打包失败的解决方式总结

Maven打包失败原因总结如下&#xff1a; 失败原因1&#xff1a;无法使用spring-boot-maven-plugin插件 使用spring-boot-maven-plugin插件可以创建一个可执行的JAR应用程序&#xff0c;前提是应用程序的parent为spring-boot-starter-parent。 需要添加parent的包spring-boot…

QML组件

一个QML文件定义了一个独立的、顶级的QML组件。 一个QML组件就是一个模板&#xff0c;被QML运行环境解释来创建一个带有一些预定义行为的对象。 一个独立的QML组件可以运行多次来禅城多个对象&#xff0c;每个对象都可以称为该组件的实例。 例子&#xff1a; 在项目中添加一…

Redis基础入门

文章目录前言一、redis是什么&#xff1f;二、安装步骤1.下载安装包2.安装三、Redis的数据类型redis是一种高级的key-value的存储系统&#xff0c;其中的key是字符串类型&#xff0c;尽可能满足如下几点&#xff1a;字符串(String)列表(List)集合(Set&#xff0c;不允许出现重复…

MySQL面试题-索引篇

1.什么是索引 MySQL的索引是一种数据结构&#xff0c;可以用于加快数据库中数据的查询速度。索引是基于表中一个或多个列的值排序的快速查找数据结构&#xff0c;可以大大提高查询效率。MySQL支持多种类型的索引&#xff0c;如B-tree索引、哈希索引、全文索引等。 索引可以在…

【java基础】异常处理(Exception)

文章目录基本介绍异常分类抛出异常非检查型异常检查型异常捕获异常捕获单个异常捕获多个异常创建自定义异常类finally字句try-with-Resource总结基本介绍 对于一个程序&#xff0c;总是有bug的。如果我们的程序遇到一个错误就终止了&#xff0c;那么肯定是不合理&#xff0c;程…

数据爬取(urllib+BeautifulSoup)

文章目录知识点总结爬虫步骤爬虫三要素爬虫注意事项python爬取技术学习网页抓取库Urllib网页解析库Beautifulsoup案例知识点总结 爬虫是一种按照一定规则&#xff0c;自动抓取互联网上网页中的相应信息的程序或脚本。 爬虫步骤 1.需求分析 2.找到要爬取信息的网站 3.下载reque…

基于halo后台管理+Gblog-wx搭建的微信小程序

先决条件 1、已经通过docker安装了halo后台管理系统(参考:http://43.136.39.20:8090/archives/halo-build) 2、安装的halo版本为1.5.3版本。此版本的halo才能安装小程序主题并启动小程序 3、需要修改小程序文件配置 解决安装的不是1.5.3的halo 1、如果是docker安装的halo…