C ++套接字不接收传入数据包,在Python中工作
Posted
技术标签:
【中文标题】C ++套接字不接收传入数据包,在Python中工作【英文标题】:C++ socket not receiving incoming packets, works in Python 【发布时间】:2017-03-02 06:50:49 【问题描述】:我找不到类似的问题,所以就这样吧。
我有一个 Linux 机器,它通过 10GbE 电缆直接连接到另一台机器。另一端的机器正在将 8kB UDP 帧推送到我的 Linux 机器上的 NIC。我可以使用 iptraf 查看传入的 UDP 数据包,并确认它们到达了预期的 NIC、IP 地址和端口号。
我可以通过以下 sn-p 使用 Python 打开一个套接字并读取此数据:
import socket
ip = '192.168.4.5'
port = 60001
sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
sock.bind((ip, port))
data = sock.recv(8192)
除了 Python 太慢之外,这没有给我带来任何问题,无论我的循环有多紧,我最终都会丢失网络数据包。数据通过 10GbE 链路传入,并且正在全速运行。
所以克服这个延迟问题,我决定转向用 C++/C 编写数据记录器。
#include <sys/types.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <string.h>
#include <netdb.h>
#include <stdio.h>
#include <arpa/inet.h>
#include <iostream>
void error(const char *msg)
perror(msg);
exit(0);
int main(int argc, char *argv[])
int sock;
ssize_t n;
socklen_t length;
in_port_t port;
const char *ip_addr;
socklen_t fromlen;
struct sockaddr_in server;
struct sockaddr_in from;
char buf[8192];
ip_addr = "192.168.4.5"; // host to listen on
port = 60001; // port to listen on
std::cout << "ip address: " << ip_addr << std::endl;
std::cout << "port: " << port << std::endl;
// create inet socket
sock = socket(AF_INET, SOCK_DGRAM, 0);
if (sock < 0) error("Opening socket");
else std::cout << "socket opened.\n";
length = sizeof(server);
memset(&server, 0, length); // initialize with zeros
server.sin_family = AF_INET;
server.sin_addr.s_addr = inet_addr(ip_addr);
server.sin_port = port;
// bind to socket address and port
if (bind(sock,(struct sockaddr *)&server, length)<0)
error("binding");
else std::cout << "binding successful\n";
// check binding status
struct sockaddr_in socket_address;
socklen_t socket_addr_len;
for (int i=0; i<10; i++)
std::cout << "iter: " << i << std::endl;
n = recv(sock, buf, 9000, 0);
if (n<0) error("recv");
std::cout << "received a datagram: ";
for (int k=0; k<10; k++)
std::cout << buf[k];
std::cout << std::endl;
return 0;
上面的 C 程序永远挂在recv()
上。如果我将其设为非阻塞,则返回EAGAIN
,表示没有可读取的内容。我尝试使用getsockname
来确定套接字是否实际绑定到正确的地址;是的。
当 Python 版本从套接字读取没有问题并且我可以直观地确认有数据传入时,我会遇到这个问题,这对我来说似乎很奇怪。我的猜测是我在 C 中的套接字设置存在问题。
另一个可能相关的信息是我的 Linux 机器在同一子网上还有 1 个其他接口。
Linux 机器上的接口
eth1:192.168.4.1 eth2: 192.168.4.5远程机器上的接口
eth0: 192.168.4.10来自 iptraf 的示例输出
UDP (8220 bytes) from 192.168.4.10:10000 to 192.168.4.5:60001 on eth2
任何指导将不胜感激。
【问题讨论】:
端口值的字节序如何? 收到 UDP 数据包时是否发送了一些“端口不可达”消息? 如果我假设您不应该使用 recv 而是使用 recvfrom 或 recvmsg,因为您使用的是 UDP,我会错吗? @Gerhardh,我刚刚编辑显示我的 eth2 实际上是 ...4.5。谢谢! @dmi:这取决于数据的使用情况。如果程序只接收数据而不发送任何响应,则可以使用 recv()。 【参考方案1】:server.sin_port = htons(端口);
【讨论】:
这让数据流动起来了!谢谢你!。我想知道这是否与@Gerhardh 关于端口字节序的评论有关。 是的。那是为了给你指明方向。以上是关于C ++套接字不接收传入数据包,在Python中工作的主要内容,如果未能解决你的问题,请参考以下文章
如何在 Linux 中创建原始套接字而不缓冲接收数据包?是不是可以?