当前位置: 首页 > news >正文

组播详解及示例代码

写在前面

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

  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;
}

相关文章:

组播详解及示例代码

写在前面 由于公司业务需要用到组播实现&#xff0c;这里就记录下学习过程。在学习组播之前&#xff0c;我们先来看看另外两种数据包传输方式&#xff1a;单播和广播。 单播&#xff1a;简单来说就是数据一对一发送&#xff0c;如果需要给多个主机发送数据时&#xff0c;就需…...

C语言-qsort函数示例解析

一.qsort函数是什么stdlib.h头文件下的函数qsort()函数&#xff1a;是八大排序算法中的快速排序&#xff0c;能够排序任意数据类型的数组其中包括整形&#xff0c;浮点型&#xff0c;字符串甚至还有自定义的结构体类型。qsort函数实现对不同元素的排序主要就是通过对compar函数…...

一些Linux内核内存性能调优笔记!

前言 在工作生活中&#xff0c;我们时常会遇到一些性能问题&#xff1a;比如手机用久了&#xff0c;在滑动窗口或点击 APP 时会出现页面反应慢、卡顿等情况&#xff1b;比如运行在某台服务器上进程的某些性能指标&#xff08;影响用户体验的 PCT99 指标等&#xff09;不达预期…...

【JVM】逃逸分析

开发者都知道&#xff0c;基本上所有对象都是在堆上创建。但是&#xff0c;这里还是没有把话说绝对哈&#xff0c;指的是基本上所有。昨天一位朋友在聊天中&#xff0c;就说了所有对象都在堆中创建&#xff0c;然后被朋友一阵的嘲笑。 开始我们的正文&#xff0c;我们今天来聊聊…...

C51---震动传感器控制LED灯亮灭

1.example #include "reg52.h" sbit led1 P3^7;//原理图中led1指向P3组IO口的P3.7口 sbit vibrate P3^3;//Do接到了P3.3口 void Delay3000ms() //11.0592MHz { unsigned char i, j, k; //_nop_(); i 22; j 3; k 227; do { …...

使用 JaCoCo 生成测试覆盖率报告

0、为什么要生成测试覆盖率报告 在我们实际的工作中&#xff0c;当完成程序的开发后&#xff0c;需要提交给测试人员进行测试&#xff0c;经过测试人员测试后&#xff0c;代码才能上线到生产环境。 有个问题是&#xff1a;怎么能证明程序得到了充分的测试&#xff0c;程序中所…...

windows下neo4j安装及配置,并绘制人物关系图谱

neo4j安装及配置&#xff0c;绘制人物关系图谱 先升级pip&#xff0c;安装py2neo pip install py2neo2021.0.1依赖 jdk1.8&#xff0c; neo4j 3.xx&#xff1b; 或者jdk18&#xff0c;neo4j 4.x&#xff0c;5.x&#xff1b; 官网下载了neo4j4.x,5.x 因为jdk版本原因都不行&am…...

【Spring6】IoC容器之基于XML管理Bean

3、容器&#xff1a;IoC IoC 是 Inversion of Control 的简写&#xff0c;译为“控制反转”&#xff0c;它不是一门技术&#xff0c;而是一种设计思想&#xff0c;是一个重要的面向对象编程法则&#xff0c;能够指导我们如何设计出松耦合、更优良的程序。 Spring 通过 IoC 容…...

Warshall算法求传递闭包及Python编程的实现

弗洛伊德算法-Floyd(Floyd-Warshall)-求多源最短路径&#xff0c;求传递闭包 Floyd算法又称为插点法&#xff0c;是一种利用动态规划的思想寻找给定的加权图中多源点之间最短路径的算法&#xff0c; 与Dijkstra算法类似。该算法名称以创始人之一、1978年图灵奖获得者、斯坦福大…...

AcWing第 93 场周赛

4867. 整除数 给定两个整数 n,k&#xff0c;请你找到大于 n 且能被 k 整除的最小整数 x。 输入格式 共一行&#xff0c;包含两个整数 n,k。 输出格式 输出大于 n 且能被 k 整除的最小整数 x。 数据范围 前 4 个测试点满足 1≤n,k≤100。 所有测试点满足 1≤n,k≤109。 …...

计及需求响应的粒子群算法求解风能、光伏、柴油机、储能容量优化配置(Matlab代码实现)

&#x1f468;‍&#x1f393;个人主页&#xff1a;研学社的博客&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5;&#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密…...

利用Nginx给RStudio-Server配置https

前篇文档&#xff0c;我这边写了安装RStudio-Server的方法。默认是http的访问方式&#xff0c;现在我们需要将其改成https的访问方式。 1、给服务器安装Nginx&#xff1a;参照之前的安装Nginx的方法。 2、创建/usr/local/nginx/ssl目录&#xff1a; mkdir /usr/local/nginx/ss…...

YOLOv7实验记录

这篇博客主要记录博主在做YOLOv7模型训练与测试过程中遇到的一些问题。 首先我们需要明确YOLO模型权重文件与模型文件的使用 其实在github的readme中已经告诉我们使用方法&#xff0c;但我相信有很多像博主一样眼高手低的人可能会犯类似的错误。 训练 首先是训练时的设置&…...

用Python获取史瓦西时空中克氏符的分量

文章目录三维球面坐标史瓦西时空三维球面坐标 Einsteinpy中提供了克氏符模型&#xff0c;可通过ChristoffelSymbols获取。简单起见&#xff0c;先以最直观的三维球面为例&#xff0c;来用Einsteinpy查看其克氏符的表达形式。 三维球面的度规张量可表示为 g001g11r2g22r2sin⁡…...

QML编码约定

QML中的国际化&#xff1a; QML使用以下函数来将字符串标记为可翻译的 qsTr()qsTranslate()qsTrld()QT_TR_NOOP()QT_TRANSLATE_NOOP()QT_TRID_NOOP最常用的还是qsTr&#xff08;&#xff09; string qsTr&#xff08;string sourceText&#xff0c; string disambiguation&…...

【Linux】安装Linux操作系统具体步骤

1). 选择创建新的虚拟机 2). 选择"典型"配置 3). 选择"稍后安装操作系统(S)" 4). 选择"Linux"操作系统,"CentOS7 64位"版本 5). 设置虚拟机的名称及系统文件存放路径 6). 设置磁盘容量 7). 自定义硬件信息 8). 启动上述创建的新虚拟机…...

前端ES6异步编程技术——Promise使用

Promise是什么 官方的定义是&#xff1a;Promise是ES6新推出的用于进行异步编程的解决方案&#xff0c;旧方案是单纯使用回调函数来解决的。对于开发人员来说&#xff0c;我们把promise当作一个普通的对象即可&#xff0c;使用它可以用来封装一个异步操作并可以获取其成功/失败…...

Kotlin实现简单的学生信息管理系统

文章目录一、实验内容二、实验步骤1、页面布局2、数据库3、登录活动4、增删改查三、运行演示四、实验总结五、源码下载一、实验内容 根据Android数据存储的内容&#xff0c;综合应用SharedPreferences和SQLite数据库实现一个用户信息管理系统&#xff0c;强化对SharedPreferen…...

413. 等差数列划分

413. 等差数列划分 如果一个数列 至少有三个元素 &#xff0c;并且任意两个相邻元素之差相同&#xff0c;则称该数列为等差数列。 例如&#xff0c;[1,3,5,7,9]、[7,7,7,7] 和 [3,-1,-5,-9] 都是等差数列。 给你一个整数数组 nums &#xff0c;返回数组 nums 中所有为等差数…...

设计模式七大原则

一、设计模式概念 1.1 软件设计模式的产生背景 "设计模式"最初并不是出现在软件设计中&#xff0c;而是被用于建筑领域的设计中。 1977年美国著名建筑大师、加利福尼亚大学伯克利分校环境结构中心主任克里斯托夫亚历山大&#xff08;Christopher Alexander&#x…...

【Mybatis系列】Mybatis常见的分页方法以及源码理解

Mybatis-Plus的selectPage 引入依赖 <dependency><groupId>com.baomidou</groupId><artifactId>mybatis-plus-boot-starter</artifactId><version>3.5.1</version></dependency>添加分页插件 Configuration public class My…...

Java面向对象:多态特性的学习

本文介绍了Java面向对象多态特性, 多态的介绍. 多态的实现条件–1.发生继承.2.发生重写(重写与重载的区别)3.向上转型与向下转型.4.静态绑定和动态绑定5. 实现多态 举例总结多态的优缺点 避免在构造方法内调用被重写的方法… Java面向对象:多态特性的学习一.什么是多态?二.多态…...

id函数 / 可变类型变量 / 不可变类型变量 / +=操作

前言 再说正文之前&#xff0c;需要大家先了解一下对象&#xff0c;指针和引用的含义&#xff0c;不懂得同学可以参考我上一篇博客“(12条消息) 引用是否有地址的讨论的_xx_xjm的博客-CSDN博客” 正文 一&#xff1a;python中一切皆对象 “python中一切皆对象”这句话我相信…...

aws apigateway 使用apigateway集成lambda

参考资料 代理集成&#xff0c;https://docs.aws.amazon.com/zh_cn/apigateway/latest/developerguide/api-gateway-create-api-as-simple-proxy-for-lambda.html非代理集成&#xff0c;https://docs.aws.amazon.com/zh_cn/apigateway/latest/developerguide/getting-started-…...

Linux SPI 驱动实验

目录 一、Linux 下 SPI 驱动框架简介 1、SPI 主机驱动 2、SPI 设备驱动 SPI 设备数据收发处理流程 3、SPI 设备和驱动匹配过程 二、添加SPI 设备信息 1、添加 ICM20608 所使用的 IO 2、 在 ecspi3 节点追加 icm20608 子节点 三、编写 ICM20608 驱动 1、修改makefile​…...

[1.4]计算机系统概述——操作系统的体系结构

第一章 计算机系统概述 操作系统的体系结构 大内核/单内核/宏内核微内核 通过之前的学习&#xff0c;我们知道计算机系统的层次结构是这样的。 但是操作系统的内部其实还可以再进一步地划分。 一部分是内核的功能&#xff0c;一部分是非内核的功能。 操作系统最核心的功能&…...

FPGA的GigE Vision IP相机图像采集方案设计,转换为千兆UDP,支持10G MAC

1 概述 GigE Vision是一个比较复杂的协议&#xff0c;要在FPGA中完全实现具有较大的难度。如果FPGA作为接收端希望实现GigE Vision相机的配置和图像采集功能&#xff0c;则只需要实现其中小部分功能即可。本文对原有GigE Vision协议的结构进行了裁剪&#xff0c;仅保留设备搜索…...

大数据相关面试题

linux 常见linux高级命令&#xff1f; top、iotopnetstatdf -hjmap -heaptarrpmps -efshell 用过的shell工具&#xff1f; awk Awk 命令详解 - 简书 awk是行处理器: 相比较屏幕处理的优点&#xff0c;在处理庞大文件时不会出现内存溢出或是处理缓慢的问题&#xff0c;通常用来…...

AI绘画第二步,抄作业复现超赞的效果!

上一篇&#xff0c;讲了如何安装AI绘画软件&#xff0c;但是装完后发现生成效果很渣&#xff01;而网上那些效果都很赞。真的是理想很丰满&#xff0c;现实很骨感。今天就是来聊聊如何抄作业&#xff0c;最大程度的还原那些超赞的效果。换一种说法就是&#xff0c;教大家如何使…...

Python的并发编程

我们将一个正在运行的程序称为进程。每个进程都有它自己的系统状态&#xff0c;包含内存状态、打开文件列表、追踪指令执行情况的程序指针以及一个保存局部变量的调用栈。通常情况下&#xff0c;一个进程依照一个单序列控制流顺序执行&#xff0c;这个控制流被称为该进程的主线…...