当前位置: 首页 > 知识库问答 >
问题:

Boost 的 UDP 异步客户端接收自己的数据报

盛琪
2023-03-14

我正在尝试使用Boost::asio进行简单的UDP客户端/服务器测试。我已经阅读了留档和官方教程,但我仍然无法正确使用。

我的问题特别是在客户端。下面是我希望客户端做的:它必须向运行在localhost:12345的服务器发送一个简单的数据报,然后侦听服务器发送的响应数据报。但这不是我得到的。看起来客户端正在发送数据报,但是在收到自己的数据报之后!在有人问“不”之前,服务器没有运行:-)

函数< code>send_datagram()似乎正确地发送了数据报,但是< code > async _ receive _ from()似乎是由发送的数据报触发的。下面是客户端的完整源代码:

#include <boost/array.hpp>
#include <boost/asio/error.hpp>
#include <boost/asio.hpp>
#include <boost/bind.hpp>
#include <boost/format.hpp>
#include <iostream>

using boost::asio::ip::udp;

const char *SERVER_ADDRESS = "localhost";
const char *SERVER_PORT = "12345";

boost::asio::io_service io_service;

udp::resolver resolver(io_service);
udp::resolver::query query(udp::v4(), SERVER_ADDRESS, SERVER_PORT);
udp::endpoint server_endpoint = (udp::endpoint) *resolver.resolve(query);
udp::endpoint remote_endpoint;
boost::asio::ip::udp::socket client_socket(io_service, server_endpoint);
boost::array<char, 2048> recv_buffer;

boost::shared_ptr<std::string> message(new std::string("test"));

void handle_recv(const boost::system::error_code& error, std::size_t /* bytes_transferred */) {

    if (error)
        std::cout << error.message() << std::endl;
    else {
        std::cout << "Datagram received" << std::endl;
    }
}

void handle_send(boost::shared_ptr<std::string> /* message */, const boost::system::error_code& error, std::size_t /* bytes_transferred */) {

    if (error)
        std::cout << error.message() << std::endl;
    else {
        std::cout << "Datagram sent" << std::endl;
    }
}

void recv_datagram() {

    client_socket.async_receive_from(
            boost::asio::buffer(recv_buffer),
            remote_endpoint,
            boost::bind(&handle_recv, boost::asio::placeholders::error, boost::asio::placeholders::bytes_transferred)
    );
}

void send_datagram() {

    client_socket.async_send_to(
            boost::asio::buffer(*message),
            server_endpoint,
            boost::bind(&handle_send, message, boost::asio::placeholders::error, boost::asio::placeholders::bytes_transferred)
    );
}

int main(int argc, char** argv) {

    std::cout << "Server address: " << server_endpoint.address() << std::endl;
    std::cout << "Server port: " << server_endpoint.port() << std::endl;
    std::cout << "Remote address: " << remote_endpoint.address() << std::endl;
    std::cout << "Remote port: " << remote_endpoint.port() << std::endl;

    send_datagram();
    recv_datagram();

    io_service.run();

    std::cout << "Done" << std::endl;

    return 0;
}

这是我得到的输出:

Server address: 127.0.0.1
Server port: 12345
Remote address: 0.0.0.0
Remote port: 0
Datagram sent
Datagram received
Done

同样,服务器在我的测试期间没有运行(事实上,它甚至还没有实现!)。

我想我正在做一件非常愚蠢的事情,但是我仍然没有弄清楚那是什么。

共有1个答案

湛功
2023-03-14

是的,所以我发现了我的问题:

boost::asio::ip::udp::socket client_socket(io_service, server_endpoint);

我不小心将客户端套接字绑定到服务器终结点,因此它有效地在端口 12345 上运行。当我调用 async_receive_from() 时,套接字正在侦听端口 12345,该端口应该运行服务器。

因此,解决方案是简单地定义一个新endpoint并将其传递到套接字结构中:

udp::endpoint local_endpoint;
boost::asio::ip::udp::socket client_socket(io_service, local_endpoint);

现在输出为:

Server address: 127.0.0.1
Server port: 12345
Remote address: 0.0.0.0
Remote port: 0
Datagram sent

现在看起来它正在正确等待服务器响应。

为了试图澄清发生了什么,当创建客户端套接字时,它必须绑定到某个本地端口。当不带参数创建local_endpoint时,它会生成一个endpoint,指向没有特定网络接口的端口0。以下是$BOOST_ROOT/BOOST/asio/ip/detail/impl/endpoint.ipp中的构造函数实现:

endpoint::endpoint()
  : data_()
{
  data_.v4.sin_family = BOOST_ASIO_OS_DEF(AF_INET);
  data_.v4.sin_port = 0;
  data_.v4.sin_addr.s_addr = BOOST_ASIO_OS_DEF(INADDR_ANY);
}

当套接字构造函数接收到这个特殊的endpoint时,它会要求操作系统选择一个随机端口,以便套接字可以绑定到。要查看那是什么端口:

std::cout << "Local address: " << client_socket.local_endpoint().address() << std::endl;
std::cout << "Local port: " << client_socket.local_endpoint().port() << std::endl;

输出示例:

Local address: 0.0.0.0
Local port: 59908
 类似资料:
  • 我正在尝试使用UDP在服务器(在公共IP上)和客户端(通过NAT)之间进行双向通信。我的逻辑是,如果服务器向IP和它接收数据包的端口发送一些数据,客户端仍然应该得到它,因为NAT最终会有将数据包发送到客户端的映射。 客户端有 2 个进程,一个用于发送数据包,另一个进程用于接收数据。服务器继续等待数据,如果它获得数据,它会将数据发送回从接收数据的端口和 IP。 客户代码如下: client_recv

  • 我是一个使用python进行套接字编程的初学者。我正在做我的课程项目。我的项目的一部分需要用不同的端口发送和接收UDP消息。提供了名为robot的服务器程序,我需要编写名为student的客户端程序,它可以与机器人进行交互。因此,我不能显示服务器程序中的所有源代码。 这是服务器程序中与UDP套接字相关的部分 这是我的客户端程序。s3 是 UDP 套接字。我可以成功地向服务器程序发送消息,但无法从中

  • 我想做的是: 为了学习游戏编程,我使用专用服务器和UDP创建了一个回声设置。专用服务器在不同的城市(也就是不在我的本地网络中)。 在我的本地计算机上,我有一个udp客户端和服务器(两个不同的程序)。当我第一次启动python服务器时,windows防火墙立即询问我是否要添加异常。在允许我的python服务器联网之后,我仍然没有收到任何消息。(客户- 我的问题是: 游戏如何解决这个问题?我没有为我想

  • 我在实现UDP连接时遇到了麻烦,因为当我在局域网内尝试它时,它是有效的,但是当NAT内部的人试图连接到公共服务器地址时,它会失败,因为从服务器作为响应发送的数据包永远不会到达客户端。 我的协议如下: 客户端A向服务器发送一个字节作为连接请求 服务器B为客户端创建一个新的套接字,并从那里向recvfrom()调用中报告的客户端端口响应一个字节。永远不会联系到客户 我也试过: 执行许多调用,每个调用在

  • 我正在测试WiFi UDP发送和接收字符串示例Arduino代码与Arduino Wifi屏蔽(最新固件更新2013年3月),我得到的行为我无法解释。 Arduino在其本地任意端口上从路由器上的另一台计算机接收UDP数据包。它通过将它自己的UDP数据包发送回远程IP:远程端口来向客户端确认它已经收到数据包。 netcat将发送UDP数据包,Arduino接收,但是netcat没有收到Arduin