通过socket获取和解析udp的导航数据

在 C++ 中获取 UDP 数据,核心流程是:创建 Socket -> 绑定端口(Bind) -> 循环接收(Recvfrom) -> 按协议解析

由于导航数据通常是结构化的(如二进制或十六进制),建议直接映射到 struct

C++ 示例代码(Linux/Windows 通用逻辑)

这里以 Linux 环境为例,展示如何读取一个假设的导航数据包:

cpp 复制代码
#include <iostream>
#include <vector>
#include <cstring>
#include <arpa/inet.h>
#include <sys/socket.h>
#include <unistd.h>

// 根据设备的通讯协议手册(报文格式)
// 1. 根据你的导航设备协议定义结构体 (注意对齐)
#pragma pack(push, 1) 
struct NavData {
    uint32_t header;    // 包头
    double lon;         // 经度
    double lat;         // 纬度
    float heading;      // 航向角
    // ... 其他字段
};
#pragma pack(pop)

int main() {
    int port = 8080; // 组合导航发送数据的目标端口
    int sockfd;
    char buffer[1024];

    // 2. 创建 UDP Socket
    if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
        perror("Socket creation failed");
        return -1;
    }

    struct sockaddr_in servaddr;
    memset(&servaddr, 0, sizeof(servaddr));
    servaddr.sin_family = AF_INET;
    servaddr.sin_addr.s_addr = INADDR_ANY; // 监听所有网卡
    servaddr.sin_port = htons(port);

    // 3. 绑定端口
    if (bind(sockfd, (const struct sockaddr *)&servaddr, sizeof(servaddr)) < 0) {
        perror("Bind failed");
        return -1;
    }

    std::cout << "Waiting for Nav Data on port " << port << "..." << std::endl;

    while (true) {
        struct sockaddr_in cliaddr;
        socklen_t len = sizeof(cliaddr);
        
        // 4. 接收数据
        int n = recvfrom(sockfd, buffer, sizeof(buffer), 0, (struct sockaddr *)&cliaddr, &len);
        
        if (n > 0) {
            // 5. 解析数据 (假设收到的就是 NavData 结构)
            if (n >= sizeof(NavData)) {
                NavData* data = reinterpret_cast<NavData*>(buffer);
                
                // 打印结果
                std::cout << "Lon: " << data->lon << " Lat: " << data->lat 
                          << " Heading: " << data->heading << std::endl;
            }
        }
    }

    close(sockfd);
    return 0;
}

关键点解析:

  1. #pragma pack(1) : 组合导航通常发送的是原始二进制流。这个指令防止 C++ 编译器在结构体中插入填充字节,确保内存布局与传感器发送的 协议文档 完全一致。
  2. INADDR_ANY : 如果你的电脑有多个网卡(比如网线连导航,WiFi上网),使用 INADDR_ANY 可以接收发往任何网卡的数据。
  3. 大端/小端 (Endianness) : 多数现代导航设备(及 X86 电脑)使用小端模式。如果数据乱码,请检查协议是否要求使用 ntohlntohs 进行字节序转换。
  4. 缓冲区大小 : buffer 应略大于设备发送的最长数据包。
相关推荐
Shingmc32 分钟前
【Linux】网络层(IP协议)
网络·网络协议·tcp/ip
Yana.nice2 分钟前
rpm -K检查RPM软件包完整性
linux·服务器·网络
Bat U6 分钟前
JavaEE|网络原理HTTP和HTTPS
网络协议·http·https
夏日听雨眠9 小时前
LInux(逻辑地址与物理地址的区别,文件描述符,lseek函数)
linux·运维·网络
ydyd2026042110 小时前
制造业数字化干货:设备巡检、报修、保养一体化管理流程拆解
网络
Hali_Botebie11 小时前
【图卷积网络】GCN是AXΘ 和CNN是AX
网络·人工智能·cnn
IpdataCloud11 小时前
高并发场景下IP数据接口怎么选?从QPS到离线库的完整选型指南
网络·网络协议·tcp/ip
CableTech_SQH12 小时前
企业园区网络突然中断排查时间影响生产?综合布线运维管理解决方案分析
网络
難釋懷12 小时前
Redis网络模型-IO多路复用模型-poll模式
网络·数据库·redis
ALINX技术博客12 小时前
【黑金云课堂】FPGA技术教程FPGA基础:呼吸灯实验+RAM/ROM IP设计与验证
网络协议·fpga开发·fpga