通过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 应略大于设备发送的最长数据包。
相关推荐
聊点儿技术19 小时前
IP风险等级评估是什么?跨境电商业务场景全解析
网络·网络协议·tcp/ip
herinspace19 小时前
如何解决管家婆辉煌零售POS中显示的原价和售价不一致?
网络·人工智能·学习·excel·语音识别·零售
JS_SWKJ19 小时前
多网闸级联部署避坑指南:安全与性能如何兼得?
网络·安全
Lyyaoo.19 小时前
【JAVA网络面经】网络模型(OSI+TCP/IP)
网络
路溪非溪20 小时前
网络运输层:TCP协议详解(一)
网络·网络协议·tcp/ip
汽车仪器仪表相关领域20 小时前
Kvaser Leaf Light HS v2 M12:5 针 M12 NMEA 2000 接口,海事与工业 CAN 总线测试的防水耐用之选
大数据·网络·人工智能·功能测试·安全性测试
爱吃芹菜炒肉20 小时前
Chapter 16: Power Management
服务器·c语言·网络·tcp/ip·pcie
运维行者_21 小时前
通过OpManager的Windows服务监控能力释放最佳IT网络性能
服务器·开发语言·网络·windows·web安全·php
爱学习的小囧21 小时前
ESXi性能历史怎么监控?2种方法,图形化+命令行全覆盖
java·linux·运维·服务器·网络·esxi·esxi8.0
小草儿7991 天前
gbase8s之onatpe备份与恢复性能测试
linux·服务器·网络