You need to enable JavaScript to run this app.
最新活动
大模型
产品
解决方案
定价
生态与合作
支持与服务
开发者
了解我们

使用recvfrom()和MSG_PEEK过滤UDP数据包及特定对等方接收问询

问题解答

1. 带MSG_PEEKrecvfrom()能否获取源地址?

可以。recvfrom()无论是否设置MSG_PEEK标志,只要传入有效的sockaddr_storage*socklen_t*参数,内核都会把UDP数据包的源地址填充到对应的结构体中。MSG_PEEK仅控制数据包是否从接收队列中移除,不会影响源地址的获取。

你可以在UdpPeer::recv()中这样实现地址校验逻辑:

ssize_t UdpPeer::recv(int servFd, std::string &msg, size_t n, int flags) {
    struct sockaddr_storage srcAddr;
    socklen_t srcAddrLen = sizeof(srcAddr);
    char tempBuf[1024]; // 临时缓冲区大小按需调整

    // 先通过MSG_PEEK获取源地址和数据(数据保留在队列)
    ssize_t peekLen = recvfrom(servFd, tempBuf, sizeof(tempBuf), flags | MSG_PEEK, 
                               (struct sockaddr*)&srcAddr, &srcAddrLen);
    if (peekLen < 0) {
        if (errno == EAGAIN || errno == EWOULDBLOCK) {
            return -1; // 非阻塞模式下无数据,正常返回
        }
        return peekLen; // 传递其他错误
    }

    // 对比源地址与当前存储的remoteAddr
    if (srcAddrLen != addrlen || memcmp(&remoteAddr, &srcAddr, srcAddrLen) != 0) {
        // 地址不匹配,直接接收丢弃数据包(取消MSG_PEEK)
        recvfrom(servFd, tempBuf, sizeof(tempBuf), flags & ~MSG_PEEK, 
                 (struct sockaddr*)&srcAddr, &srcAddrLen);
        return -1; // 标记跳过了非目标地址的数据包
    }

    // 地址匹配,正常接收数据到msg
    msg.resize(n);
    ssize_t recvLen = recvfrom(servFd, msg.data(), n, flags, nullptr, nullptr);
    if (recvLen > 0) {
        msg.resize(recvLen);
    }
    return recvLen;
}

2. 更优的UDP特定对等方消息接收方案

手动PEEK校验地址的方式可行,但有更高效的替代方案:

使用connect()绑定UDP套接字到特定对等方

UDP虽是无连接协议,但调用connect()可以让套接字与指定远程地址绑定:

  • 绑定后,send()/recv()可直接使用(无需每次指定地址),内核只会向绑定地址发送数据,且仅接收来自该地址的数据包,其他地址的数据包会被内核直接丢弃。
  • 这种方式省去了用户态的地址校验逻辑,避免了额外的recvfrom()调用,性能更优。

示例代码片段:

// 在UdpPeer中添加connect方法
void UdpPeer::connect(int servFd) {
    int ret = ::connect(servFd, (struct sockaddr*)&remoteAddr, addrlen);
    if (ret < 0) {
        throw std::system_error(errno, std::generic_category());
    }
}

// 简化后的recv方法
ssize_t UdpPeer::recv(int servFd, std::string &msg, size_t n, int flags) {
    msg.resize(n);
    ssize_t recvLen = ::recv(servFd, msg.data(), n, flags);
    if (recvLen > 0) {
        msg.resize(recvLen);
    }
    return recvLen;
}

两种方案对比

  • 手动校验地址:灵活性高,适合同时处理多个对等方的场景,但需要额外系统调用,性能略低。
  • connect绑定:内核层面过滤数据包,性能更高、代码更简洁,适合仅与单个特定对等方通信的场景。若对等方地址变化,重新调用connect()更新绑定即可。

内容的提问来源于stack exchange,提问作者pastor

火山引擎 最新活动