组播例子

本文介绍了一个简单的组播通信实现方案,包括发送端(sender)和接收端(receiver)的C++代码示例。通过设置socket选项并使用特定的网络接口,演示了如何在局域网内进行组播数据包的发送与接收。

1.sender.cpp

#include <sys/types.h> 
#include <sys/socket.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include <unistd.h>
#include <stdio.h>
#include <string.h>
#include <net/if.h>
#include <iostream>
using namespace std;
int main(int agrc, char** argv) {
	int fd = socket(AF_INET, SOCK_DGRAM, 0);
	if (fd < 0) {
		cerr << "socket created failed...!" << endl;
		return -1;
	}
	int port = atoi(argv[1]);
	struct sockaddr_in addr = {0};
	addr.sin_family = AF_INET;
	addr.sin_port = htons(port);
	addr.sin_addr.s_addr = htonl(INADDR_ANY);
	if (bind(fd, (struct sockaddr*)&addr, sizeof(addr)) < 0) {
		close(fd);
		cerr << "bind failed on port = " << port << endl;
		return -1;
	}
	struct ip_mreqn n = {0};
	inet_pton(AF_INET, "192.168.4.10", &n.imr_multiaddr.s_addr);	// 组播地址
	inet_pton(AF_INET, "0.0.0.0", &n.imr_address.s_addr);			// 本机地址
	n.imr_ifindex = if_nametoindex("enp8s0");						//本机的通信用的网卡的物理地址,可以用ifconfig查看
  	int ret = setsockopt(fd, IPPROTO_IP, IP_MULTICAST_IF, &n, sizeof(n));
	if (ret < 0) {
		close(fd);
		cerr << "set multicast socket failed...!" << endl;
		return -1;
	}
	struct sockaddr_in client = {0};
	client.sin_family = AF_INET;
	client.sin_port = htons(6666);
	inet_pton(AF_INET, "192.168.4.10", &client.sin_addr.s_addr);
	char buf[64] = "";
	int cnt = 0;
	while (true) {
		sprintf(buf, "count = %d\n", cnt++);
		int ret = sendto(fd, buf, sizeof buf, 0, (struct sockaddr*)&client, sizeof(client));
		if (-1 == ret){
			perror("sendto -1");
		}
		sleep(1);
	}
  	close(fd);

	return 0;
}

2.receiver.cpp

#include <sys/types.h> 
#include <sys/socket.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include <unistd.h>
#include <stdio.h>
#include <string.h>
#include <net/if.h>
#include <iostream>
using namespace std;

int main(int agrc, char** argv) {
	int fd = socket(AF_INET, SOCK_DGRAM, 0);
	if (fd < 0) {
		cerr << "socket created failed...!" << endl;
        return -1;
    }
	struct sockaddr_in addr = {0};
	addr.sin_family = AF_INET;
	addr.sin_port = htons(6666);
	inet_pton(AF_INET, "0.0.0.0", &addr.sin_addr.s_addr);
	socklen_t len = sizeof(addr);
	if (bind(fd, (struct sockaddr*)&addr, sizeof(addr)) < 0) {
		close(fd);
        cerr << "bind failed on port = 6666" << endl;
        return -1;
	}
	struct ip_mreqn n = {0};
	inet_pton(AF_INET, "192.168.4.10", &n.imr_multiaddr.s_addr);
	inet_pton(AF_INET, "0.0.0.0", &n.imr_address.s_addr);
	n.imr_ifindex = if_nametoindex("enp8s0");
	char buf[64] = "";
	int ret =  setsockopt(fd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &n, sizeof(n));
	while (true) {
		int ret = recvfrom(fd, buf, sizeof(buf), 0, NULL, NULL);
		write(STDOUT_FILENO, buf, ret);
		sleep(1);
	}
	close(fd);

	return 0;
}

3.make.sh

g++ -std=c++17 -g -o SendTest sender.cpp -pthread
g++ -std=c++17 -g -o RecvTest receiver.cpp -pthread

 

<think>我们将在Qt4.8.6下编写一个UDP例子。由于Qt4.8.6不支持一些Qt5的新特性,我们将使用旧式的信号槽连接,并且注意相关的API在Qt4中的使用方式。主要步骤:1.创建一个QUdpSocket对象。2.绑定到一个端口和地址(注意需要特定的地址)。3.加入。4.接收数据:通过readyRead信号连接到槽函数,读取数据报。5.发送数据:使用writeDatagram发送到地址和端口。注意:地址范围是224.0.0.0到239.255.255.255。我们将编写一个简单的发送和接收程序。程序可以同时发送和接收消息。由于Qt4.8.6不支持lambda表达式,我们将使用传统的信号槽连接方式。代码结构:-创建一个类MulticastUDP,用于封装的发送和接收。 -在main函数中,创建该类的实例,并演示发送和接收。注意:在Qt4中,加入使用QUdpSocket::joinMulticastGroup,但要注意,在绑定之后调用。我们将编写一个简单的示例,每隔一段时间发送一条消息,并接收显示。由于Qt4.8.6没有QTimer::singleShot的lambda支持,我们将使用一个QTimer并连接其timeout信号到一个槽函数来发送消息。以下是代码:</think>### Qt 4.8.6 UDP 示例 (Linux/C++) 以下是一个完整的 UDP 示例,适用于 Linux 平台下的 Qt 4.8.6: #### `multicast.h` 头文件 ```cpp #ifndef MULTICAST_H #define MULTICAST_H #include <QObject> #include <QUdpSocket> #include <QTimer> class Multicast : public QObject { Q_OBJECT public: explicit Multicast(QObject *parent = 0); ~Multicast(); void startReceiver(const QString &multicastGroup, quint16 port); void startSender(const QString &multicastGroup, quint16 port, const QString &message); private slots: void onReadyRead(); void sendMessage(); private: QUdpSocket *m_socket; QTimer *m_timer; QString m_multicastGroup; quint16 m_port; QString m_message; }; #endif // MULTICAST_H ``` #### `multicast.cpp` 实现文件 ```cpp #include "multicast.h" #include <QHostAddress> #include <QNetworkInterface> #include <QDebug> #include <sys/socket.h> #include <netinet/in.h> #include <arpa/inet.h> Multicast::Multicast(QObject *parent) : QObject(parent), m_socket(0), m_timer(0) { } Multicast::~Multicast() { if (m_socket) { m_socket->close(); delete m_socket; } if (m_timer) { m_timer->stop(); delete m_timer; } } void Multicast::startReceiver(const QString &multicastGroup, quint16 port) { if (m_socket) return; m_socket = new QUdpSocket(this); m_multicastGroup = multicastGroup; m_port = port; // 绑定到端口 if (!m_socket->bind(port, QUdpSocket::ShareAddress)) { qWarning() << "Failed to bind to port" << port; return; } // 加入 QHostAddress groupAddress(multicastGroup); if (!groupAddress.isMulticast()) { qWarning() << multicastGroup << "is not a multicast address"; return; } if (!m_socket->joinMulticastGroup(groupAddress)) { qWarning() << "Failed to join multicast group" << multicastGroup; return; } // 设置TTL(可选) int ttl = 1; // 限制在本地网络 int sockfd = m_socket->socketDescriptor(); if (sockfd != -1) { setsockopt(sockfd, IPPROTO_IP, IP_MULTICAST_TTL, &ttl, sizeof(ttl)); } // 连接信号槽 connect(m_socket, SIGNAL(readyRead()), this, SLOT(onReadyRead())); qDebug() << "Listening for multicast on" << multicastGroup << ":" << port; } void Multicast::startSender(const QString &multicastGroup, quint16 port, const QString &message) { if (m_socket) return; m_socket = new QUdpSocket(this); m_multicastGroup = multicastGroup; m_port = port; m_message = message; // 创建定时器用于发送消息 m_timer = new QTimer(this); connect(m_timer, SIGNAL(timeout()), this, SLOT(sendMessage())); m_timer->start(2000); // 每2秒发送一次 qDebug() << "Sending multicast to" << multicastGroup << ":" << port; } void Multicast::onReadyRead() { while (m_socket->hasPendingDatagrams()) { QByteArray datagram; datagram.resize(m_socket->pendingDatagramSize()); QHostAddress sender; quint16 senderPort; m_socket->readDatagram(datagram.data(), datagram.size(), &sender, &senderPort); qDebug() << "Received from" << sender.toString() << ":" << senderPort << "=>" << QString::fromUtf8(datagram); } } void Multicast::sendMessage() { QByteArray datagram = m_message.toUtf8(); QHostAddress groupAddress(m_multicastGroup); // 获取所有网络接口 QList<QNetworkInterface> interfaces = QNetworkInterface::allInterfaces(); foreach (const QNetworkInterface &interface, interfaces) { if (interface.flags() & QNetworkInterface::CanMulticast) { // 设置输出接口 int sockfd = m_socket->socketDescriptor(); if (sockfd != -1) { struct in_addr ifaceAddr; QList<QNetworkAddressEntry> entries = interface.addressEntries(); if (!entries.isEmpty()) { QHostAddress addr = entries.first().ip(); if (addr.protocol() == QAbstractSocket::IPv4Protocol) { ifaceAddr.s_addr = htonl(addr.toIPv4Address()); setsockopt(sockfd, IPPROTO_IP, IP_MULTICAST_IF, &ifaceAddr, sizeof(ifaceAddr)); } } } // 发送消息 qint64 sent = m_socket->writeDatagram(datagram, groupAddress, m_port); if (sent == -1) { qWarning() << "Failed to send multicast to" << m_multicastGroup << ":" << m_port; } else { qDebug() << "Sent multicast:" << m_message; } } } } ``` #### `main.cpp` 主程序 ```cpp #include <QCoreApplication> #include <QStringList> #include <QTimer> #include "multicast.h" int main(int argc, char *argv[]) { QCoreApplication a(argc, argv); // 参数处理 QStringList args = QCoreApplication::arguments(); if (args.size() < 4) { qDebug() << "Usage:" << args[0] << "<mode> <multicast_group> <port> [message]"; qDebug() << "Modes: receiver | sender"; qDebug() << "Example:"; qDebug() << " " << args[0] << "receiver 239.255.43.21 45454"; qDebug() << " " << args[0] << "sender 239.255.43.21 45454 \"Hello Multicast!\""; return 1; } QString mode = args[1]; QString multicastGroup = args[2]; quint16 port = args[3].toUShort(); QString message = (args.size() > 4) ? args[4] : "Default multicast message"; Multicast multicast; if (mode == "receiver") { multicast.startReceiver(multicastGroup, port); } else if (mode == "sender") { multicast.startSender(multicastGroup, port, message); } else { qWarning() << "Invalid mode:" << mode; return 1; } // 退出计时器(仅用于演示) QTimer::singleShot(30000, &a, SLOT(quit())); return a.exec(); } ``` #### `.pro` 项目文件 ```qmake QT += core network TARGET = multicast_example TEMPLATE = app CONFIG += console CONFIG -= app_bundle SOURCES += main.cpp \ multicast.cpp HEADERS += multicast.h ``` ### 编译与运行 1. **编译程序**: ```bash qmake-qt4 make ``` 2. **启动接收端**: ```bash ./multicast_example receiver 239.255.43.21 45454 ``` 3. **启动发送端** (在另一个终端): ```bash ./multicast_example sender 239.255.43.21 45454 "Hello from Qt4 multicast!" ``` ### 关键功能说明 1. **地址选择**: - 使用 `239.255.43.21` 作为示例地址(范围 224.0.0.0 到 239.255.255.255) - 避免使用 224.0.0.0-224.0.0.255(保留地址) 2. **网络接口选择**: ```cpp QList<QNetworkInterface> interfaces = QNetworkInterface::allInterfaces(); foreach (const QNetworkInterface &interface, interfaces) { if (interface.flags() & QNetworkInterface::CanMulticast) { // 使用支持的接口 } } ``` 3. **TTL (Time To Live) 设置**: ```cpp int ttl = 1; // 限制在本地网络 setsockopt(sockfd, IPPROTO_IP, IP_MULTICAST_TTL, &ttl, sizeof(ttl)); ``` 4. **指定输出接口**: ```cpp struct in_addr ifaceAddr; ifaceAddr.s_addr = htonl(addr.toIPv4Address()); setsockopt(sockfd, IPPROTO_IP, IP_MULTICAST_IF, &ifaceAddr, sizeof(ifaceAddr)); ``` ### 常见问题解决 1. **"joinMulticastGroup failed" 错误**: - 确保网络接口支持(`ifconfig` 查看) - 检查防火墙设置:`sudo ufw allow 45454/udp` - 尝试绑定到特定接口: ```cpp m_socket->joinMulticastGroup(groupAddress, QNetworkInterface::interfaceFromName("eth0")); ``` 2. **接收不到数据**: - 验证路由:`sudo route add -net 239.255.43.21 netmask 255.255.255.255 dev eth0` - 使用 `tcpdump` 调试: ```bash sudo tcpdump -i eth0 -n udp port 45454 ``` 3. **IPv6 支持**: ```cpp // 对于IPv6 QHostAddress groupAddress("ff02::1"); // 所有节点的本地链路范围 setsockopt(sockfd, IPPROTO_IPV6, IPV6_MULTICAST_IF, ...); ``` ### 地址范围参考 | 地址范围 | 用途 | |-------------------------|--------------------------| | 224.0.0.0 - 224.0.0.255 | 本地网络控制(保留) | | 224.0.1.0 - 224.0.1.255 | 互联网控制 | | 232.0.0.0 - 232.255.255.255 | 源特定 (SSM) | | 233.0.0.0 - 233.255.255.255 | GLOP 地址(AS号映射) | | 239.0.0.0 - 239.255.255.255 | 管理范围 |
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值