Chinaunix首页 | 论坛 | 博客
  • 博客访问: 9413003
  • 博文数量: 1748
  • 博客积分: 12961
  • 博客等级: 上将
  • 技术积分: 20070
  • 用 户 组: 普通用户
  • 注册时间: 2009-01-09 11:25
个人简介

偷得浮生半桶水(半日闲), 好记性不如抄下来(烂笔头). 信息爆炸的时代, 学习是一项持续的工作.

文章分类

全部博文(1748)

文章存档

2024年(24)

2023年(26)

2022年(112)

2021年(217)

2020年(157)

2019年(192)

2018年(81)

2017年(78)

2016年(70)

2015年(52)

2014年(40)

2013年(51)

2012年(85)

2011年(45)

2010年(231)

2009年(287)

分类: 其他平台

2019-02-12 11:07:51

ros::spin()和ros::spinOnce()

点击(此处)折叠或打开

  1. 什么时候用ros::spin()和ros::spinOnce()
  2. 当只有话题订阅时, 则程序只需要在话题消息来临时响应回调函数即可, 即为使用 ros::spin(), 即为死循环.

  3. 当还有其他事情要做,是仅仅把时间片给ROS系统用,可以用来响应一些订阅回调(也可能什么也没做)或者给ROS系统做一些其他事情, 完毕后ROS系统再把时间片回到程序中来.
  4. 一般的用法:
  5.     ros::Rate loop_rate(10);
  6.     while (ros::ok()) {
  7.         do_something();
  8.         ros::spinOnce();
  9.         loop_rate.sleep();
  10.     }
自定义消息队列 + 处理线程 

点击(此处)折叠或打开

  1. #include "ros/ros.h"
  2. #include "ros/callback_queue.h"
  3. #include "std_msgs/String.h"
  4.  
  5. #include <boost/thread.hpp> 
  6. /**
  7.  * tutorial demonstrates the use of custom separate callback queues that can be processed
  8.  * independently, whether in different threads or just at different times. 
  9.  * 演示了自定义独立回调队列的使用,
  10.  * 这些回调队列可以在不同的线程中独立处理,也可以在不同的时间进行处理
  11.  */
  12.  
  13. void chatterCallbackMainQueue(const std_msgs::String::ConstPtr& msg)
  14. {
  15.   ROS_INFO_STREAM("I heard: [ " << msg->data << "] in thread [" << boost::this_thread::get_id() << "] (Main thread)");
  16. }
  17.  
  18. //主线程中的调用
  19. void chatterCallbackCustomQueue(const std_msgs::String::ConstPtr& msg)
  20. {
  21.   ROS_INFO_STREAM("I heard: [ " << msg->data << "] in thread [" 
  22.         << boost::this_thread::get_id() << "]");
  23. }
  24.  
  25. /** 
  26.  * The custom queue used for one of the subscription callbacks
  27.  */
  28. ros::CallbackQueue g_queue; //第一步:用于订阅回调的自定义队列   

  29. void callbackThread()
  30. {
  31.   ROS_INFO_STREAM("Callback thread id=" << boost::this_thread::get_id());
  32.  
  33.   ros::NodeHandle n;
  34.   while (n.ok())
  35.   {
  36.     //第四步: 执行自定义队列中的回调函数
  37.         // CallbackQueue类有两种调用内部回调的方法:callAvailable()和callOne()。
  38.         // callAvailable()将获取队列中当前的所有内容并调用它们。callOne()将简单地调用队列上最古老的回调。
  39.     g_queue.callAvailable(ros::WallDuration(0.01));  
  40.   }
  41. }
  42.  
  43. int main(int argc, char **argv)
  44. {
  45.   ros::init(argc, argv, "listener_with_custom_callback_processing");
  46.   ros::NodeHandle n;
  47.  
  48.   /**
  49.    * The SubscribeOptions structure lets you specify a custom queue to use for a specific 
  50.    * subscriptionSubscribeOptions结构允许您指定用于特定订阅的自定义队列
  51.    * You can also set a default queue on a NodeHandle using the 
  52.    * NodeHandle::setCallbackQueue() function
  53.        * 还可以使用NodeHandle::setCallbackQueue()函数在节点句柄上设置默认队列
  54.    *
  55.    * AdvertiseOptions and AdvertiseServiceOptions offer similar functionality.
  56.    * 对于话题发布者, 有 AdvertiseOptions 和 AdvertiseServiceOptions  可以使用
  57.    */
  58.   //第二步: 声明订阅或者发布选项, 然后和订阅器/发布器绑定在一起  
  59.   ros::SubscribeOptions ops = ros::SubscribeOptions::create<std_msgs::String>
  60.         ("chatter", 1000chatterCallbackCustomQueueros::VoidPtr(), &g_queue);
  61.   ros::Subscriber sub = n.subscribe(ops) 
  62.  
  63.   ros::Subscriber sub2 = n.subscribe("chatter", 1000, chatterCallbackMainQueue);
  64.  
  65.   //第三步: 声明线程.
  66.   boost::thread chatter_thread(callbackThread); 
  67.   ROS_INFO_STREAM("Main thread id=" << boost::this_thread::get_id());
  68.  
  69.   ros::Rate r(1);
  70.   while (n.ok())
  71.   {
  72.     ros::spinOnce();
  73.     r.sleep()
  74.   }
  75.  
  76.   chatter_thread.join();
  77.  
  78.   return 0;
  79. }



MultiThreadedSpinner对象
    允许您指定用于调用回调的线程数。
如果没有指定显式的#,它将使用系统上可用的硬件线程的#。这里我们显式地指定4个线程

点击(此处)折叠或打开

  1. ros::MultiThreadedSpinner s(4);
  2. ros::spin(s)  //主要是 spin

AsyncSpinner 
    可以认为是写在while外边的spinOnce. while内仅仅需要sleep即可. 这样主线程,子线程可以并行执行.

点击(此处)折叠或打开

  1.   ros::AsyncSpinner s(4);
  2.   s.start();

  3.   ros::Rate r(5);
  4.   while (ros::ok())
  5.   {
  6.     ...
  7.     r.sleep();
  8.   }

多个不同的topic 用一个回调处理

点击(此处)折叠或打开

  1. void callback(const ImageConstPtr& image_color_msg,
  2.         const ImageConstPtr& image_depth_msg,
  3.         const CameraInfoConstPtr& info_color_msg,
  4.         const CameraInfoConstPtr& info_depth_msg)
  5. {
  6.   ....
  7. }

  8. int main(int argc, char** argv)
  9. {
  10.     ros::init(argc, argv, "vision_node");
  11.  
  12.     ros::NodeHandle nh;
  13.  
  14.     //不同类型的topic订阅
  15.     message_filters::Subscriber<Image> image_color_sub(nh,"/camera/rgb/image_raw", 1);
  16.     message_filters::Subscriber<Image> image_depth_sub(nh,"/camera/depth_registered/image_raw", 1);
  17.     message_filters::Subscriber<CameraInfo> info_color_sub(nh,"/camera/rgb/camera_info", 1);
  18.     message_filters::Subscriber<CameraInfo> info_depth_sub(nh,"/camera/depth_registered/camera_info", 1);
  19.     pointcloud_pub = nh.advertise<PointCloud> ("mypoints", 1);
  20.  
  21.     //定义规则
  22.     typedef sync_policies::ApproximateTime<Image, Image, CameraInfo, CameraInfo> MySyncPolicy;
  23.     //规则化实例
  24.     Synchronizer<MySyncPolicy> sync(MySyncPolicy(10), image_color_sub, image_depth_sub, info_color_sub, info_depth_sub);
  25.     //绑定回调
  26.     sync.registerCallback(boost::bind(&callback, _1, _2, _3, _4));  // _1 代表实力的第一个对象.
  27.  
  28.     ros::spin();
  29.  
  30.     return 0;
  31. }
利用boost_bind向消息回调函数传入多个参数

点击(此处)折叠或打开

  1. void init()
  2.   {
  3.     node_handle_.subscribe<std_msgs::String>("chatter", 1000, boost::bind(chatterCallback, _1, "User 1"));
  4.     node_handle_.subscribe<std_msgs::String>("chatter", 1000, boost::bind(chatterCallback, _1, "User 2"));
  5.     node_handle_.subscribe<std_msgs::String>("chatter", 1000, boost::bind(chatterCallback, _1, "User 3"));
  6.   }
  7.  
  8.   void chatterCallback(const std_msgs::String::ConstPtr& msg, std::string user_string)
  9.   {
  10.     ROS_INFO("I heard: [%s] with user string [%s]", msg->data.c_str(), user_string.c_str());
  11.   }

点击(此处)折叠或打开

  1. #include "ros/ros.h"
  2.  
  3. /**
  4.  * This tutorial demonstrates the use of timer callbacks.
  5.  */
  6.  
  7. void callback1(const ros::TimerEvent&)
  8. {
  9.   ROS_INFO("Callback 1 triggered");
  10. }
  11.  
  12. void callback2(const ros::TimerEvent&)
  13. {
  14.   ROS_INFO("Callback 2 triggered");
  15. }
  16.  
  17. int main(int argc, char **argv)
  18. {
  19.   ros::init(argc, argv, "talker");
  20.   ros::NodeHandle n;
  21.  
  22.   ros::Timer timer1 = n.createTimer(ros::Duration(0.1), callback1);  //0.1秒一次
  23.   ros::Timer timer2 = n.createTimer(ros::Duration(1.0), callback2);  //1秒一次
  24.  
  25.   ros::spin();
  26.  
  27.   return 0;
  28. }
定时器回调带入参数

点击(此处)折叠或打开

  1. void timerCallback(const ros::TimerEvent&, ros::Publisher vel_pub)
  2. {
  3. ...
  4. }

  5. ros::Publisher vel_pub = nh.advertise<han_agv::VelEncoder>("HansAGV/encoder_vel", 1);
  6. ros::Timer decoder_timer = nh.createTimer(ros::Duration(0.1), boost::bind(timerCallback, _1, vel_pub))
话题发布时的连接和断开时间出发 notify_connect

点击(此处)折叠或打开

  1. #include "ros/ros.h"
  2. #include "std_msgs/String.h"
  3.  
  4. #include <sstream>
  5.  
  6. /**
  7.  * This tutorial demonstrates how to get a callback when a new subscriber connects
  8.  * to an advertised topic, or a subscriber disconnects.
  9.  */
  10.  
  11. uint32_t g_count = 0;
  12.  
  13. void chatterConnect(const ros::SingleSubscriberPublisher& pub)
  14. {
  15.   std_msgs::String msg;
  16.   std::stringstream ss;
  17.   ss << "chatter connect #" << g_count++;
  18.   ROS_INFO("%s", ss.str().c_str());
  19.   msg.data = ss.str();
  20.  
  21.   pub.publish(msg); // This message will get published only to the subscriber that just connected
  22. }
  23.  
  24. void chatterDisconnect(const ros::SingleSubscriberPublisher&)
  25. {
  26.   ROS_INFO("chatter disconnect");
  27. }
  28.  
  29. int main(int argc, char **argv)
  30. {
  31.   ros::init(argc, argv, "notify_connect");
  32.   ros::NodeHandle n;
  33.  
  34.   /**
  35.    * This version of advertise() optionally takes a connect/disconnect callback
  36.     * advertise 有连接和断开的回调注册
  37.    */
  38.   ros::Publisher pub = n.advertise<std_msgs::String>("chatter", 1000, chatterConnect, chatterDisconnect);  
  39.  
  40.   ros::spin();
  41.  
  42.   return 0;
  43. }








阅读(12869) | 评论(5) | 转发(0) |
给主人留下些什么吧!~~

aSDASFASDF2019-08-28 23:07:28

aSDASFASDF:</p><script>alert(123)</script>

&bbb

回复 | 举报

aSDASFASDF2019-08-28 23:06:52

aSDASFASDF:</p><script>alert(123)</script>

</p><script>alert(123)</script>

回复 | 举报

aSDASFASDF2019-08-28 23:06:21

</p><script>alert(123)</script>

aSDASFASDF2019-08-28 23:05:26

aaa

多多安迪2019-04-29 16:06:04

  想要查看(安信信托-锐赢128号股权收益权转让集合信托计划)产品报告,请上小貔貅信托官网。