首页 > 解决方案 > 访问/插入值时,Boost strands post 导致分段错误

问题描述

我目前正在编写一个网络结构,它只是让服务器监听它自己的链,很简单。每当调用 strand 以调用函数并且该函数从类中调用任何成员时,都会发生分段错误。

有一些例外,但值得注意的是,这些分段错误发生在

  1. tcp::acceptor myAcceptor(my_io_service, myEndpoint)
  2. mySockets.insert(...)

现在我可以轻松地交换 myAcceptor 并将其放在构造函数的初始化列表中,并将其视为类成员,但我想避免这种情况。

那么接下来要验证我的声明呢?我最近的尝试是检查 mySockets 的大小,我应该注意,它是一个map。我在调用 myStrand.post(...)之前检查了大小,并在函数调用中检查了大小。这是我注意到的。

mySockets 映射大小:

  1. 发布之前:0(没有插入任何东西,太棒了!)
  2. 在函数调用中,通过邮寄方式调用:999929272 (Ughhhh)

看起来有些东西很通风,但究竟是什么?它是否创建了一个新实例,但奇怪的是这不是 0 吗?它不可能丢失地址,因为这肯定会导致分段错误,不是吗?

这就是我现在在调试过程中所处的位置。

目标

最终目标是能够在 post 链中访问 mySockets 和 my_io_service。

布局流程

这解释了布局,好的 ol 代码如下

  1. 通过传递 io_service 创建“工作”
  2. 通过调用 .run() 启动 io_service
  3. 创建 TCP 服务器
  4. 为类成员类型分配参数
  5. 调用 myStrand.post(...) 来调用服务器,监听连接的客户端
  6. 等待客户端连接等

我选择了详细说明问题的代码示例。

头文件

using boost::asio::ip::tcp;
bool amIConnected = false;
boost::asio::io_service::strand myStrand;
boost::asio::io_service& my_io_service;
std::map<std::string, std::shared_ptr<tcp::socket>> mySockets;
tcp::acceptor myAcceptor

源文件

ConstructorStuff::ConstructorStuff(boost::asio::io_service& the_service) : 
myStrand(the_service), 
my_io_service(the_service),
myAcceptor(the_service, tcp::endpoint(tcp::v4(), 1337))
{}

void ConstructorStuff::CreateTheServer()
{
    amIConnected = true;
    // std::cout << mySockets.size() << std::endl; Returns 0
    mStrand.post(std::bind(&ConstructorStuff::ListenForConnection, this));
}

void ConstructorStuff::ListenForConection()
{
    // std::cout << mySockets.size() << std::endl; Crazy size
    tcp::acceptor tryAgain(my_io_service, tcp::endpoint(tcp::v4(), 1338)); // Just to showcase the error doesn't just occur through mySockets.
}

在哪里调用 ConstructorStuff

void SetupServer()
{
    ConstructorStuff stuff(some_service);
    stuff.CreateTheServer();
}

这调用了上面的函数

ServiceStarter spinup(service);
spinup.Start(); // Just queues reference_service.run()
std::shared_ptr<SomeClass> server_interface(new SomeClass ( service ));
server_interface->SetupServer();
boost::this_thread::sleep(boost::posix_time::seconds(20));
server_interface->Cancel(); // Just another function that kills the server.

任何建议都会很棒!

旁注

编辑

我添加了两个新的源文件。我只能展示我所能展示的内容,但这应该足以说明问题。

标签: c++c++11postboostasio

解决方案


问题出SetupServer在哪里stuff是局部变量,所以在CreateTheServer被调用之后,被销毁,并且由处理程序发布的stuff某处被删除的对象调用。然后打印出奇怪的结果——这是一个未定义行为的简单示例。io_service::runmStrand.post(std::bind(&ConstructorStuff::ListenForConnection, this));std::cout << mySockets.size() << std::endl;


推荐阅读