UDP的组播发送与接收C语言测试和nc接收组播测试

2024-06-14 17:12

本文主要是介绍UDP的组播发送与接收C语言测试和nc接收组播测试,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

组播这个东西,很多年前用过一次。本身的原理不复杂,未知的是使用的环境,受使用环境的影响有多大,还是那句废话,具体问题具体分析。

发送端代码multicast.c

#include <stdio.h>  
#include <stdlib.h>  
#include <string.h>  
#include <unistd.h>  
#include <sys/types.h>  
#include <sys/socket.h>  
#include <netinet/in.h>  
#include <arpa/inet.h>  
#include <netdb.h>  #define MULTICAST_ADDR "224.0.0.1" // 组播地址  
#define MULTICAST_PORT 9990       // 组播端口  int main(int argc, char *argv[]) {  int sockfd;  struct sockaddr_in local,addr;  char message[] = "Hello, multicast!";  // 1. 创建UDP套接字  if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {  perror("socket creation failed");  exit(EXIT_FAILURE);  }  // 2. 设置套接字选项以允许组播  int reuse = 1;  if (setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof(reuse)) < 0) {  perror("setsockopt (SO_REUSEADDR) failed");  exit(EXIT_FAILURE);  }int yes = 1;  if (setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, &yes, sizeof(int)) == -1) {  perror("setsockopt");  exit(1);  }  // 3. 绑定套接字(可选,通常不需要)  // ...  local.sin_family = AF_INET;  // 假设你想绑定到本地的IP地址 "192.168.1.100",你可以通过inet_addr或inet_pton来设置  local.sin_addr.s_addr = inet_addr("192.168.0.3");  // 或者使用inet_pton来处理IPv6地址  // if (inet_pton(AF_INET, "192.168.0.3", &serv_addr.sin_addr) <= 0) {  //     perror("inet_pton failed");  //     exit(EXIT_FAILURE);  // }  local.sin_port = htons(12345); // 假设端口是12345  // 4. 构造组播地址结构  memset(&addr, 0, sizeof(addr));  addr.sin_family = AF_INET;  addr.sin_addr.s_addr = inet_addr(MULTICAST_ADDR);  addr.sin_port = htons(MULTICAST_PORT);  // 5. 发送数据  if (sendto(sockfd, message, strlen(message), 0, (const struct sockaddr *)&addr, sizeof(addr)) < 0) {  perror("sendto failed");  exit(EXIT_FAILURE);  }  // 6. 接收数据(如果需要)  // ...  // 7. 关闭套接字  close(sockfd);  return 0;  
}

接收端代码multicast_recv.c

#include <stdio.h>  
#include <stdlib.h>  
#include <string.h>  
#include <unistd.h>  
#include <sys/types.h>  
#include <sys/socket.h>  
#include <netinet/in.h>  
#include <arpa/inet.h>  
#include <netdb.h>  #define MULTICAST_ADDR "224.0.0.1" // 组播地址  
#define MULTICAST_PORT 9990       // 组播端口  
#define BUFFER_SIZE 1024            // 接收缓冲区大小  int main(int argc, char *argv[]) {  int sockfd;  struct sockaddr_in addr;  char buffer[BUFFER_SIZE];  struct ip_mreq mreq;  socklen_t addrlen = sizeof(addr);  // 1. 创建UDP套接字  if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {  perror("socket creation failed");  exit(EXIT_FAILURE);  }  int yes = 1;  if (setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, &yes, sizeof(int)) == -1) {  perror("setsockopt");  exit(1);  }// 2. 设置组播地址和端口  memset(&addr, 0, sizeof(addr));  addr.sin_family = AF_INET;  addr.sin_addr.s_addr = htonl(INADDR_ANY); // 绑定到所有可用的网络接口  addr.sin_port = htons(MULTICAST_PORT);  // 3. 绑定套接字(如果需要特定的端口,就绑定;否则,通常不需要绑定)  if (bind(sockfd, (struct sockaddr *)&addr, sizeof(addr)) < 0) {  perror("bind failed");  exit(EXIT_FAILURE);  }  // 4. 设置组播选项  mreq.imr_multiaddr.s_addr = inet_addr(MULTICAST_ADDR);  mreq.imr_interface.s_addr = htonl(INADDR_ANY); // 或者设置为特定的网络接口IP  if (setsockopt(sockfd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &mreq, sizeof(mreq)) < 0) {  perror("setsockopt failed");  exit(EXIT_FAILURE);  }  // 5. 接收数据  printf("Waiting for multicast packets...\n");  while (1) {  int nbytes = recvfrom(sockfd, buffer, BUFFER_SIZE, 0, (struct sockaddr *)&addr, &addrlen);  if (nbytes < 0) {  perror("recvfrom failed");  exit(EXIT_FAILURE);  }  buffer[nbytes] = '\0'; // 确保字符串以null字符结尾  printf("Received: %s\n", buffer);  }  // 6. 关闭套接字(注意:由于我们在一个无限循环中,这行代码实际上不会被执行)  close(sockfd);  return 0;  
}

编译脚本:

#!/bin/bashecho "hello"
source /opt/fsl-imx-xwayland/5.4-zeus/environment-setup-aarch64-poky-linux
echo $ARCH
aarch64-poky-linux-gcc --sysroot=/opt/fsl-imx-xwayland/5.4-zeus/sysroots/aarch64-poky-linux multicast.c -o multicast
gcc multicast_recv.c -o multicast_recv
sudo cp multicast /home/lkmao/nfsroot/yocto/home/root/
#make
exit 0

设置网关:

测试发现,如果不设置网关,组播数据发不出去。

 route add default gw 192.168.0.1 dev eth1

root@imx8mpevk:~# route add default gw 192.168.0.1 dev eth1
root@imx8mpevk:~# route -n
Kernel IP routing table
Destination     Gateway         Genmask         Flags Metric Ref    Use Iface
0.0.0.0         192.168.0.1     0.0.0.0         UG    0      0        0 eth1
0.0.0.0         0.0.0.0         0.0.0.0         U     0      0        0 eth0
169.254.0.0     0.0.0.0         255.255.0.0     U     0      0        0 eth0
192.168.0.0     0.0.0.0         255.255.255.0   U     0      0        0 eth1
root@imx8mpevk:~#

测试结果

使用nc作为接收端测试

测试的时候,发现,直接监听udp的9990端口就可以,这个有点诧异啊。

nc -ul 9990

组播地址查看

ip maddr show  

开发板的地址: 

root@imx8mpevk:~# ip maddr show dev eth1
3:      eth1link  33:33:00:00:00:01link  01:00:5e:00:00:01link  33:33:ff:07:0b:a5link  33:33:00:00:02:02link  33:33:00:00:00:fblink  01:00:5e:00:00:fbinet  224.0.0.251inet  224.0.0.1inet6 ff02::fbinet6 ff02::202inet6 ff02::1:ff07:ba5inet6 ff02::1inet6 ff01::1
root@imx8mpevk:~#

ubuntu的广播查询:

lkmao@lkmao-virtual-machine:~$ ip maddr show dev ens33
2:      ens33link  01:00:5e:00:00:01link  33:33:00:00:00:01link  33:33:ff:9a:b7:5alink  01:00:5e:00:00:fblink  33:33:00:00:00:fbinet  224.0.0.251inet  224.0.0.1inet6 ff02::fbinet6 ff02::1:ff9a:b75ainet6 ff02::1inet6 ff01::1
lkmao@lkmao-virtual-machine:~$

 根据执行命令可知,默认的组播组都是224.0.0.1,这也说明了,为什么直接执行nc -ul 9990可以接收到组播数据。

小结

这个还得具体问题具体分析吧。

这篇关于UDP的组播发送与接收C语言测试和nc接收组播测试的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1061021

相关文章

使用SQL语言查询多个Excel表格的操作方法

《使用SQL语言查询多个Excel表格的操作方法》本文介绍了如何使用SQL语言查询多个Excel表格,通过将所有Excel表格放入一个.xlsx文件中,并使用pandas和pandasql库进行读取和... 目录如何用SQL语言查询多个Excel表格如何使用sql查询excel内容1. 简介2. 实现思路3

Go语言实现将中文转化为拼音功能

《Go语言实现将中文转化为拼音功能》这篇文章主要为大家详细介绍了Go语言中如何实现将中文转化为拼音功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 有这么一个需求:新用户入职 创建一系列账号比较麻烦,打算通过接口传入姓名进行初始化。想把姓名转化成拼音。因为有些账号即需要中文也需要英

Go语言使用Buffer实现高性能处理字节和字符

《Go语言使用Buffer实现高性能处理字节和字符》在Go中,bytes.Buffer是一个非常高效的类型,用于处理字节数据的读写操作,本文将详细介绍一下如何使用Buffer实现高性能处理字节和... 目录1. bytes.Buffer 的基本用法1.1. 创建和初始化 Buffer1.2. 使用 Writ

深入理解C语言的void*

《深入理解C语言的void*》本文主要介绍了C语言的void*,包括它的任意性、编译器对void*的类型检查以及需要显式类型转换的规则,具有一定的参考价值,感兴趣的可以了解一下... 目录一、void* 的类型任意性二、编译器对 void* 的类型检查三、需要显式类型转换占用的字节四、总结一、void* 的

C语言线程池的常见实现方式详解

《C语言线程池的常见实现方式详解》本文介绍了如何使用C语言实现一个基本的线程池,线程池的实现包括工作线程、任务队列、任务调度、线程池的初始化、任务添加、销毁等步骤,感兴趣的朋友跟随小编一起看看吧... 目录1. 线程池的基本结构2. 线程池的实现步骤3. 线程池的核心数据结构4. 线程池的详细实现4.1 初

如何测试计算机的内存是否存在问题? 判断电脑内存故障的多种方法

《如何测试计算机的内存是否存在问题?判断电脑内存故障的多种方法》内存是电脑中非常重要的组件之一,如果内存出现故障,可能会导致电脑出现各种问题,如蓝屏、死机、程序崩溃等,如何判断内存是否出现故障呢?下... 如果你的电脑是崩溃、冻结还是不稳定,那么它的内存可能有问题。要进行检查,你可以使用Windows 11

详解Spring Boot接收参数的19种方式

《详解SpringBoot接收参数的19种方式》SpringBoot提供了多种注解来接收不同类型的参数,本文给大家介绍SpringBoot接收参数的19种方式,感兴趣的朋友跟随小编一起看看吧... 目录SpringBoot接受参数相关@PathVariable注解@RequestHeader注解@Reque

Java如何接收并解析HL7协议数据

《Java如何接收并解析HL7协议数据》文章主要介绍了HL7协议及其在医疗行业中的应用,详细描述了如何配置环境、接收和解析数据,以及与前端进行交互的实现方法,文章还分享了使用7Edit工具进行调试的经... 目录一、前言二、正文1、环境配置2、数据接收:HL7Monitor3、数据解析:HL7Busines

SpringBoot中Get请求和POST请求接收参数示例详解

《SpringBoot中Get请求和POST请求接收参数示例详解》文章详细介绍了SpringBoot中Get请求和POST请求的参数接收方式,包括方法形参接收参数、实体类接收参数、HttpServle... 目录1、Get请求1.1 方法形参接收参数 这种方式一般适用参数比较少的情况,并且前后端参数名称必须

性能测试介绍

性能测试是一种测试方法,旨在评估系统、应用程序或组件在现实场景中的性能表现和可靠性。它通常用于衡量系统在不同负载条件下的响应时间、吞吐量、资源利用率、稳定性和可扩展性等关键指标。 为什么要进行性能测试 通过性能测试,可以确定系统是否能够满足预期的性能要求,找出性能瓶颈和潜在的问题,并进行优化和调整。 发现性能瓶颈:性能测试可以帮助发现系统的性能瓶颈,即系统在高负载或高并发情况下可能出现的问题