- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
您好,我是 Socket 编程的新手,我尝试创建一个客户端服务器应用程序,其中我的服务器是相机,我的 C++ 应用程序中是客户端。当我看到计算机和相机之间的数据包传输时,它显示相机在停止后发送了超过 150000 个数据包。但是,当我收到我能够一次接收 400 - 450 个数据包
时,recvfrom
函数进入等待状态。如果我在不停止前一个文件的情况下再次运行该 exe 文件,它将再次收到 400-450 个数据包
。
接收数据包代码
SOCKET out1 = socket(AF_INET, SOCK_RAW, IPPROTO_UDP);
if (out1 == INVALID_SOCKET)
{
cout << out1 << endl;
}
server.sin_family = AF_INET;
server.sin_port = htons(3956);
inet_pton(AF_INET, "192.168.1.140", &server.sin_addr);
int serverLength = sizeof(server);
connect(out1, (sockaddr*)&server, serverLength);
while (1)
{
memset(buf, 0, sizeof(buf));
int bytesIn = recvfrom(out1, buf, 1444, 0, (sockaddr*)&server, &serverLength);
if (bytesIn > 0)
{
cout << "Image Received :" << bytesIn <<packet_counter << endl;
packet_counter++;
}
else
{
cout << "Not Received : " << endl;
}
}
我正在使用管理员权限运行 .exe。
谁能告诉我为什么 recvfrom 函数会进入等待状态。
提前致谢。
编辑:-
抱歉,我提供了完整的代码。
#include <stdio.h>
#include <Windows.h>
#include <thread>
#include <WinSock2.h>
// Library
#pragma comment(lib, "ws2_32.lib")
using namespace std;
//***** Function Decleration *****//
void _packetConfig(SOCKET);
void _sendPacket(SOCKET, const char*, int, int);
// Global Variable
sockaddr_in server;
//***** Main Function *****//
void main(char argc, char* argv[])
{
WSADATA data;
WORD version = MAKEWORD(2, 2);
if(WSAStartup(version, &data) == SOCKET_ERROR)
{
cout << "Can't Start Socket" << WSAGetLastError<<endl;
return;
}
char buf[2000];
SOCKET out1 = socket(AF_INET, SOCK_RAW, IPPROTO_UDP);
if (out1 == INVALID_SOCKET)
{
cout << out1 << endl;
}
server.sin_family = AF_INET;
server.sin_port = htons(3956);
inet_pton(AF_INET, "192.168.1.140", &server.sin_addr);
int serverLength = sizeof(server);
connect(out1, (sockaddr*)&server, serverLength);
int packet_counter = 0;
SOCKET out = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);
_packetConfig(out);
cout << "Inside Main" << endl;
while (1)
{
//connect(out1, (sockaddr*)&server, serverLength);
memset(buf, 0, sizeof(buf));
int bytesIn = recvfrom(out1, buf, 1444, 0, (sockaddr*)&server, &serverLength);
if (bytesIn > 0)
{
cout << "Image Received :" << bytesIn <<packet_counter << endl;
packet_counter++;
}
else
{
cout << "Not Received : " << endl;
}
}
WSACleanup();
}
//***** Function to Send Bytes to the Camera *****//
void _sendPacket(SOCKET sock, const char* s, int len, int i)
{
int sendOk = sendto(sock, (const char*)s, len, 0, (sockaddr*)&server, sizeof(server));
if (sendOk == SOCKET_ERROR)
{
cout << "Didn't Work" << WSAGetLastError() << endl;
}
else
{
cout << "\nSend Succesfully" << " " << i << endl;
}
char buf[2000];
int serverLength = sizeof(server);
int bytesIn = recvfrom(sock, buf, 2000, 0, (sockaddr*)&server, &serverLength);
if (bytesIn > 0)
{
cout << "Message Received :" << bytesIn << endl;
}
}
//***** Function to call the _sendPacket function and send commands to the Camera *****//
void _packetConfig(SOCKET sock)
{
// 59 Commands and every command call _snedPacket function to send commands to camera it will working properly
}
在上面的代码中,我必须首先发送这个用_packetConfig
函数编写的59命令
,然后只有相机会发送我收到的图像数据包
所有命令的回复。
当我也使用该代码运行 wireshark 时,我可以看到在这 59 个命令之后相机给出 3580*51
数据包。即 51 帧,每帧包含 3580 个数据包
最佳答案
感谢您发布代码。它实际上有一些问题,所以首先我将发布一些代码作为引用,然后提及我在之后注意到的主要问题。
好的,这是一些适合我的代码:
#include <WinSock2.h> // ** before** windows.h
#include <WS2tcpip.h>
#include <iostream>
#include <stdio.h>
#include <Windows.h>
#include <assert.h>
#pragma comment (lib, "ws2_32.lib")
const int port = 3956;
// main
int main (char argc, char* argv[])
{
WSADATA wsadata;
WORD version = MAKEWORD(2, 2);
int err = WSAStartup (MAKEWORD (2, 2), &wsadata);
if (err)
{
std::cout << "WSAStartup failed, error: " << err << std::endl;
return 255;
}
char buf [1444];
bool send = argc > 1 && _stricmp (argv [1], "send") == 0;
if (send)
{
// Send
SOCKET skt_out = socket (AF_INET, SOCK_DGRAM, IPPROTO_UDP);
assert (skt_out != INVALID_SOCKET);
sockaddr_in destination_address = { };
destination_address.sin_family = AF_INET;
destination_address.sin_port = htons (port);
inet_pton (AF_INET, "192.168.1.2", &destination_address.sin_addr);
memset (buf, 'Q', sizeof (buf));
printf ("Sending: ");
for ( ; ; )
{
sendto (skt_out, buf, sizeof (buf), 0, (const sockaddr *) &destination_address, sizeof (destination_address));
printf (".");
Sleep (50);
}
closesocket (skt_out);
WSACleanup ();
return 0;
}
// Receive
SOCKET skt_in = socket (AF_INET, SOCK_DGRAM, IPPROTO_UDP);
assert (skt_in != INVALID_SOCKET);
int receive_buffer_size = 65536;
if ((setsockopt (skt_in, SOL_SOCKET, SO_RCVBUF, (const char *) &receive_buffer_size, sizeof (int)) ) < 0)
std::cout << "Could not set SO_RCVBUF, error: " << WSAGetLastError () << std::endl;
sockaddr_in receive_address = { };
receive_address.sin_family = AF_INET;
receive_address.sin_port = htons (port);
receive_address.sin_addr.s_addr = htonl (INADDR_ANY);
if (bind (skt_in, (const sockaddr *) &receive_address, sizeof (receive_address)) == -1)
{
std::cout << "bind failed , error: " << WSAGetLastError () << std::endl;
return 255;
}
int packetCounter = 0;
printf ("Receiving: ");
for ( ; ; )
{
int bytesIn = recvfrom (skt_in, buf, sizeof (buf), 0, NULL, 0);
if (bytesIn > 0)
std::cout << "Packet received:" << bytesIn << " bytes (" << ++packetCounter << ")" << std::endl;
else
std::cout << "Receive error: " << WSAGetLastError () << std::endl;
}
closesocket (skt_in);
WSACleanup ();
return 0;
}
要在“发送”模式下运行它,请将send 指定为命令行中的第一个参数。否则它充当接收器(也称为服务器)。
那么你的代码有什么问题?好吧,排名不分先后:
SOCK_RAW
bind
以便它知道要监听哪个端口。 sockaddr *from
参数到 recvfrom
并不意味着你认为它意味着什么(请检查 docs )。您会看到我将其作为 NULL 传递。WSAStartup
的返回值。同样,请检查文档。但是说了这么多,您实际上缺少对 bind
的调用。我重写了代码,因为你的代码比较乱。
另外,重要的细节,UDP 不保证传送 - 有很多原因导致已发送的数据包未收到或什至可能收到乱序(您的相机是否以某种方式对数据包进行排序?)
您需要在应用程序的逻辑中满足这一点(这是一个问题,最好使用 TCP,它可以保证数据包的传送和排序)。
关于c++ - recvfrom 在进入等待状态后只收到很少的数据包,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51028499/
伙计们, 这个问题困扰我好久了,求大神指点!我在一台笔记本电脑上有服务器,它正在将数据发送到另一台笔记本电脑上的客户端,同一个局域网。我可以在服务器端和客户端的 wireshark 上看到 udp 数
这是我的 udp 服务器端代码 #include"headers.h" int main(int argc,char **argv) { //------------udp server so
给定len参数设置为10240,我必须反复调用该函数以获取所有数据。但这最终导致阻塞。如何获得所有数据并以独立于平台的方式安全返回? 顺便说一句,我在发送方使用netcat: cat ocr_pi.p
我正在使用 recvfrom 从网络接收多播消息。 系统上有两个网络接口(interface),recvfrom 只从第一个接口(interface)接收。 当第 2 个接口(interface)正常
我想处理 SIGUSR1 信号,所以我做了它需要的一切并且它工作了。但是在我的程序中,我在 recvfrom 等待,当我处理信号然后返回到 recvfrom 时,我从这个函数中得到了 -1。是否可以返
这是一个多客户端套接字程序。 当我尝试运行我的程序时,最后几个recvfrom() 不返回任何数据。但之前的recvfrom有多余的数据。有人可以帮我解决这个问题吗? 客户: while (1) {
我有一个 UDP 服务器实现,我在 recvfrom 调用中遇到段错误。 #define SIZEOF sizeof #define PKTSIZE 65535 char tmp_bu
在基于客户端-服务器架构的设置中,我的客户端通过大小为 116 字节的 UDP 套接字发送数据。但在服务器端,程序只接收24个字节。如有任何帮助,我们将不胜感激。 我尝试使用wireshark捕获数据
我正在编写一个简单的 udp 服务器应用程序,当我尝试获取远程连接用户的地址时遇到错误。 while(run == 1) { struct sockaddr_in client_addr;
我最近用 C 语言编写了一个 netflow 接收器,它通过 UDP 监听 netflow。一旦处理完流数据报,它就会将其写入系统日志。我的问题是,一旦我收到带有标准 recvfrom 函数的数据报包
我有一个简单的代码来发送和接收 ICMP 数据包,例如 ping,一切正常:我的数据包已发送并且收到了结果。我的recvfrom结果有问题,缓冲区中的ip dest/src不是服务器dest ip。
我有开源 cod(这是它的链接) 访问https://github.com/jtRIPper/dns-tcp-socks-proxy/blob/master/dns_proxy.c在鳕鱼的这一部分
在准备第一次编码 UDP 时,我正在尝试一些从 here 复制并稍微修改过的示例客户端和服务器代码。 。一切似乎都正常,除了由 recvfrom() 返回的值始终是缓冲区的大小而不是读取的字节数(如果
我正在用 C 语言编写一个简单的多线程客户端-服务器 UDP 程序,其中我一次向服务器发送多条消息,并在接收线程中等待回复。我这样做是为了测试发送和接收数据包的时间。 我已经用我的服务器在本地测试了
我需要实现以下行为:当服务器启动时,它应该使用广播检查现有服务器。然后它等待回答。 但是,如何设置等待超时? int optval = 1; char buff[BUFF_SIZE]; SOCKADD
您好,我是 Socket 编程的新手,我尝试创建一个客户端服务器应用程序,其中我的服务器是相机,我的 C++ 应用程序中是客户端。当我看到计算机和相机之间的数据包传输时,它显示相机在停止后发送了超过
我在使用 recvfrom() 时遇到一些问题,无法在 C++ 中获取所有数据包。 (然后阻塞不返回) 我发送了 1 个查询数据包,然后发送了一个响应。 它被分成多个805字节的数据包,然后以一个~2
我不明白 recvfrom 在以下情况下的行为:我有一个发送和接收 udp 数据包并休眠的循环;根据该位接收 (recvfrom) 延迟会有所不同。如果我睡了 22 毫秒或更长时间,则延迟约为 170
我在编写小型 DNS 服务器并将其 strip 化到最低限度时偶然发现了一个奇怪的行为。该程序应在 127.0.0.1:1337 上监听 DNS 查询并以拒绝回复。我通过发出 dig @localho
我正在尝试通过 udp 实现距离 vector 路由。我有以下结构: struct cost_info { uint8_t id; uint8_t pad; uint16_t
我是一名优秀的程序员,十分优秀!