提升异步操作不起作用(对我来说)

时间:2014-12-01 15:58:46

标签: c++ asynchronous boost boost-asio

首先,这是我第一次使用boost :: asio以及异步编程。所以,我完全不熟悉两者中的任何一个。

基本上我想通过串口与机器人进行交互。为此,我使用boost :: asio :: serial_port。我想要做的其中一个操作是让机器人旋转几毫秒,但不同步,以免在其他处理过程中出现任何延迟。该课程的内部结构如下:

class Robot
{
    boost::asio::io_service is;
    boost::asio::serial_port port;

    ...

    public:

    Robot(const std::string &visionDeviceAddress, const std::string &motorControlDeviceAddress)
    :visionDevice(visionDeviceAddress), port(is), motorControlDevice(motorControlDeviceAddress)

    ...

    void completePendingMotions()
    {
        is.run();
    }
}

我相信以下功能应该可以胜任:

void Robot::async_rotateLeftFor(unsigned long milliseconds)
{
    boost::asio::deadline_timer t(is, boost::posix_time::milliseconds(milliseconds));
    //the character 'a' initiates a non-stop anticlockwise rotation
    char c='a';
    boost::asio::write(port, boost::asio::buffer(&c,1));
    t.async_wait([&](boost::system::error_code e)
    {
        //to stop the rotation, i need to pass the character 'q'
        //this is done synchronously by function stop()
        stop();
    });
}

最后,来自main()的调用如下:

int main(void)
{
    Robot r("0","COM6");
    r.connect();
    r.async_rotateLeftFor(2000);
    r.completePendingMotions();
    return 0;
}

我得到的只是机器人连接成功建立,它开始旋转,但它不会停止,因为它应该由于完成处理程序。我不知道可能是什么原因。缺乏关于asio的文档也无济于事。非常感谢任何帮助。

1 个答案:

答案 0 :(得分:1)

问题是deadline_timer对象需要保持活动状态,直到处理程序触发,否则在调用is.run()时会立即触发处理程序并出现错误。当async_rotateLeftFor函数退出时,计时器将被销毁。

我做的是保持计时器,将计时器对象包装在shared_ptr中并将其传递给处理程序对象。

void Robot::async_rotateLeftFor(unsigned long milliseconds) {   
    auto t = std::make_shared<boost::asio::deadline_timer>(
            is, boost::posix_time::milliseconds( milliseconds )); 

    //...

    // (capture shared_ptr in lambda)
    t->async_wait( [this,t](boost::system::error_code e )
        {  
            stop(); 
        }
    );