Qt 是一个跨平台C++图形界面开发库,利用Qt可以快速开发跨平台窗体应用程序,在Qt中我们可以通过拖拽的方式将不同组件放到指定的位置,实现图形化开发极大的方便了开发效率,本章将重点介绍如何运用QUdpSocket组件实现基于UDP的组播通信。

组播是一种一对多的通信方式,允许一个发送者将数据报文发送到多个接收者,这些接收者通过共享相同的组播IP地址进行通信。在设置组播地址时需要注意,该范围被限制在239.0.0.0~239.255.255.255以内,这是预留给组播的地址范围。

setSocketOption 设置套接字

在Qt中使用组播,首先需要调用setSocketOption函数,该函数是 QUdpSocket 类的成员函数,用于设置套接字的选项。

该函数原型如下:

bool QUdpSocket::setSocketOption(
QAbstractSocket::SocketOption option,
const QVariant & value
)
  • option:要设置的套接字选项,这里应该是 QAbstractSocket::MulticastTtlOption,表示设置多播 TTL 选项。
  • value:选项的值,这里应该是 TTL 的值。在 IPv4 中,TTL 是一个 8 位的字段,表示数据报在网络中允许经过的最大路由器数量。通常情况下,TTL 值越大,数据报能够传播的范围就越广。

函数返回一个 bool 类型的值,表示是否成功设置了选项。如果设置成功,返回 true,否则返回 false

MainWindow::MainWindow(QWidget *parent)
: QMainWindow(parent)
, ui(new Ui::MainWindow)
{
ui->setupUi(this); udpSocket=new QUdpSocket(this); // 设置为多播
udpSocket->setSocketOption(QAbstractSocket::MulticastTtlOption,1);
}

bind 绑定套接字地址

接着就是对特定端口的绑定,绑定端口可以通过调用bind函数,该函数用于将 QUdpSocket 绑定到指定的本地地址和端口,并设置特定的绑定选项。

在我们的课件中,使用 bind()QUdpSocket 绑定到 IPv4 的任意地址,并指定了一个组播(Multicast)端口,同时设置了共享地址(ShareAddress)选项。

该函数原型如下:

void QUdpSocket::bind(
const QHostAddress & address,
quint16 port,
BindMode mode = DefaultForPlatform
)
  • address:要绑定的本地地址,这里使用 QHostAddress::AnyIPv4 表示绑定到 IPv4 的任意地址。
  • port:要绑定的本地端口号,这里应该是组播端口号。
  • mode:绑定模式,指定套接字的行为。这里使用 QUdpSocket::ShareAddress 表示共享地址选项,它允许多个套接字同时绑定到相同的地址和端口。

函数将 QUdpSocket 绑定到指定的地址和端口,并且允许多个套接字同时共享相同的地址和端口。

joinMulticastGroup 加入组播

joinMulticastGroup() 函数是 QUdpSocket 类的成员函数,用于将 QUdpSocket 加入指定的多播组。

该函数原型如下:

bool QUdpSocket::joinMulticastGroup(
const QHostAddress & groupAddress,
const QNetworkInterface & iface = QNetworkInterface()
)
  • groupAddress:要加入的多播组的组播地址。
  • iface:要加入多播组的网络接口。默认情况下,会选择默认的网络接口。

函数返回一个 bool 类型的值,表示是否成功加入了多播组。如果成功加入多播组,返回 true;否则返回 false。通过调用 joinMulticastGroup() 函数,QUdpSocket 将成为指定多播组的成员,并能够接收该多播组发送的数据报。

// 开始组播
void MainWindow::on_pushButton_start_clicked()
{
// 获取IP
QString IP= ui->lineEdit_address->text();
groupAddress=QHostAddress(IP); // 获取端口
quint16 groupPort = ui->lineEdit_port->text().toUInt(); // 绑定端口
if (udpSocket->bind(QHostAddress::AnyIPv4, groupPort, QUdpSocket::ShareAddress))
{
// 加入组播
udpSocket->joinMulticastGroup(groupAddress);
ui->plainTextEdit->appendPlainText("[*] 加入组播 " + IP + ":" + QString::number(groupPort));
}
}

leaveMulticastGroup 退出组播

leaveMulticastGroup() 函数用于将 QUdpSocket 从指定的多播组中移除。通过调用该函数,QUdpSocket 将不再是指定多播组的成员,不再接收该多播组发送的数据报。

该函数原型如下:

bool QUdpSocket::leaveMulticastGroup(
const QHostAddress & groupAddress,
const QNetworkInterface & iface = QNetworkInterface()
)
  • groupAddress:要离开的多播组的组播地址。
  • iface:要离开多播组的网络接口。默认情况下,会选择默认的网络接口。

函数返回一个 bool 类型的值,表示是否成功离开了多播组。如果成功离开多播组,返回 true;否则返回 false

// 关闭组播
void MainWindow::on_pushButton_stop_clicked()
{
// 退出组播
udpSocket->leaveMulticastGroup(groupAddress);
udpSocket->abort();
ui->plainTextEdit->appendPlainText("[-] 退出组播");
}

writeDatagram 发送数据报

writeDatagram() 函数是 QUdpSocket 类的成员函数,用于发送数据报到指定的多播组。通过调用该函数,可以将数据报发送到指定的多播组和端口,让其他成员接收到该数据报。

其函数原型如下:

qint64 QUdpSocket::writeDatagram(
const QByteArray & datagram,
const QHostAddress & groupAddress,
quint16 port
)
  • datagram:要发送的数据报的内容,通常是一个 QByteArray 对象。
  • groupAddress:要发送到的多播组的组播地址。
  • port:要发送到的多播组的端口号。

函数返回一个 qint64 类型的值,表示实际发送的字节数。如果发送成功,返回发送的字节数;否则返回 -1。

// 发送组播消息
void MainWindow::on_pushButton_send_clicked()
{
quint16 groupPort = ui->lineEdit_port->text().toUInt();
QString msg=ui->lineEdit_msg->text();
QByteArray datagram=msg.toUtf8(); udpSocket->writeDatagram(datagram,groupAddress,groupPort);
}

readDatagram 接收数据报

readDatagram() 函数是 QUdpSocket 类的成员函数,用于从套接字中读取数据报,并将其存储到指定的缓冲区中。通常情况下,可以使用这个函数来接收来自其他主机的数据报。通过使用该函数可从套接字中读取数据报,并获取数据报的源地址和端口号。

其函数原型如下:

qint64 QUdpSocket::readDatagram(
char * data, qint64 maxSize,
QHostAddress * address = nullptr,
quint16 * port = nullptr
)
  • data:指向用于存储接收数据的缓冲区的指针。
  • maxSize:缓冲区的最大大小,即最多可以接收的字节数。
  • address:指向用于存储发送数据报的源地址的 QHostAddress 对象的指针。
  • port:指向用于存储发送数据报的源端口号的 quint16 类型的指针。

该函数返回一个 qint64 类型的值,表示实际接收的字节数。如果接收成功,返回接收的字节数;否则返回 -1。

// 读取数据报
void MainWindow::onSocketReadyRead()
{
while(udpSocket->hasPendingDatagrams())
{
QByteArray datagram;
datagram.resize(udpSocket->pendingDatagramSize());
QHostAddress peerAddr;
quint16 peerPort;
udpSocket->readDatagram(datagram.data(),datagram.size(),&peerAddr,&peerPort); QString str=datagram.data(); QString peer="[从 "+peerAddr.toString()+":"+QString::number(peerPort)+" 发送] "; ui->plainTextEdit->appendPlainText(peer+str);
}
}

读者可自行运行课件程序,并在多台电脑中配置相同网段,当点击发送消息时所有同网段的程序都将收到广播,如下图所示;

C++ Qt开发:QUdpSocket实现组播通信的更多相关文章

  1. IPv4组播通信原理

    2011-05-08 21:21:14 标签:组播 vin_do,vin_do学习笔记,笔记 休闲 职场 摘自网络,感谢原作者 摘要: 本文试图成为学习TCP/IP网络组播技术的入门材料.文中介绍了组 ...

  2. 【网络开发】winsock组播

    https://my.oschina.net/lopo/blog/260685 //客户端 #include <winsock2.h> #include <stdio.h> # ...

  3. Python3组播通信编程实现教程(发送者+接收者)

    一.说明 1.1 标准组播解释 通信分为单播.多播(即组播).广播三种方式 单播指发送者发送之后,IP数据包被路由器发往目的IP指定的唯一一台设备的通信形式,比如你现在与web服务器通信就是单播形式 ...

  4. zigbee组播通信原理

    组播: 在zigbee网络里面,把网络节点标记为组的方式来进行通信:发送模块如果发送的组号和网络里标记模块的组号相对应,那么这些模块就可以拿到这些无线数据包. 特点: 1.分组中组的编号有两个字节. ...

  5. 【网络开发】UDP组播接收端解析

    UDP组播接收端解析 网络中的一台主机如果希望能够接收到来自网络中其它主机发往某一个组播组的数据报,那么这么主机必须先加入该组播组,然后就可以从组地址接收数据包.在广域网中,还涉及到路由器支持组播路由 ...

  6. IP组播技术介绍及实现例子

    引 言 近年来,随着Internet的迅速普及和爆炸性发展,在Internet上产生了许多新的应用,其中不少是高带宽的多媒体应用,譬如网 络视频会议.网络音频/视频广播.AOD/VOD.股市行情发布. ...

  7. IP组播

    1  IP组播基础 IP组播技术有效地解决了单点发送.多点接收的问题.组播源只发送一份数据,被传递的信息在距组播源尽可能远的网络节点才开始被复制和分发,并且只发送给需要该信息的接收者.  说明: 本章 ...

  8. Ztack学习笔记(6)-广播组播点播

    Zigbee网络中进行数据通信主要有三种类型:单播.组播.广播.那这三种方式如何设置呢,在哪里设置呢, 一. 广播 当应用程序需要将数据包发送给网络的每一个设备时,使用这种模式.广播的短地址有三种 0 ...

  9. UDP单播和组播使用SO_REUSEADDR 测试结果

    UDP单播通信 一. 预置条件 A.B在同一台机器,网络中存在往A.B所在的机器的8888端口发送单播UDP数据 A:端口复用绑定在端口8888上 B:端口复用绑定在端口8888上操作步骤:(1)先启 ...

  10. C#Soket组播

    using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.N ...

随机推荐

  1. NC15832 Most Powerful

    题目链接 题目 题目描述 Recently, researchers on Mars have discovered N powerful atoms. All of them are differe ...

  2. Java 递归的方式将list集合的某一字段拼接单个String

    场景介绍 要将list 集合中的某一个字段合并成一个字符串,并且要用符号 "|" 分割开每个拼接后的字段. 一个例子胜于一切的文字表达,拼接后的结果如下 str1|str2|str ...

  3. 【Android】使用Binder实现进程间传递对象案例

    1 前言 使用AIDL实现进程间通讯简单案例 和 使用AIDL实现进程间传递对象案例 中介绍了使用 AIDL 进行进程间通讯,其本质仍然是Binder,aidl 文件对应生成的接口中,将服务端调用的抽 ...

  4. Java设计模式-原型模式Prototype

    介绍 当我们有一个类的实例(Prototype)并且我们想通过复制原型来创建新对象时,通常使用Prototype模式. 原型模式是一种创建型设计模式.能够复制已有对象, 而又无需使代码依赖它们所属的类 ...

  5. js数组类型

    js数组类型: 数组检测 1.判断变量是否为数组类型: arr1 instanceof Array Array.isArray(arr1); true 转换方法 toString()方法,以便返回数组 ...

  6. 启动MySQL5.7服务无法启动或Table 'mysql.plugin' doesn't exist

    首先说一下我这个是mysql5.7.16免安装版,不过这个问题对于5.7版本应该都适用. 问题重现: 安装过程也说一下吧: 1.将下载的压缩文件解压到指定目录,     我的是:E:\program\ ...

  7. Python笔记五之正则表达式

    本文首发于公众号:Hunter后端 原文链接:Python笔记五之正则表达式 这一篇笔记介绍在 Python 里使用正则表达式. 正则表达式,Regular Expression,可用于在一个目标字符 ...

  8. 万字Java进阶笔记总结

    JavaApi 字符串 String 注意:Java中"=="操作符的作用: 基本数据类型:比较的是内容. 引用数据类型比较的是对象的内存地址. StringBuffer/Stri ...

  9. maven配置全局私服地址和阿里云仓库

    直接上配置代码 <?xml version="1.0" encoding="UTF-8"?> <!-- Licensed to the Apa ...

  10. 【Azure Redis 缓存】Azure Redis服务开启了SSL(6380端口), PHP如何访问缓存呢?

    问题描述 使用6379端口连接Azure Redis服务,连接失败.因为默认情况下Azure Redis的设置没有打开6379的端口.需要使用SSL(6380端口)进行连接,但是遇见了无法连接的问题. ...