首先,这是我第一次使用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的文档也无济于事。非常感谢任何帮助。
答案 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();
}
);