阅读ROS1 noetic源码ros_comm,得到如下结论:
一、rospy
python版本的ROS1节点会给每个subscriber,每个timer新建一个线程。所以python ROS节点根本不需要rospy.spin()来执行回调,因为subscriber和timer的回调都在各自的线程里运行,rospy.spin()的作用只是阻塞主线程保证程序不退出罢了。
但要注意多线程并不意味着并发,python由于全局解释锁(GIL)的存在,不管其进程中开了几条“线程”,仍然实际只占用了系统的一条线程,不会利用cpu的多核并发性能。所以python的线程在并发角度其实时假线程。要在一个程序内实现真并发,可以用multi-process库,实际上也是开多个进程。
rospy由于每个subsciber和timer的回调都在各自单独的线程中执行,所以并不是线程安全的。也就是说你需要手动为不同回调函数中同时用到的变量进行加锁等操作来保证线程安全。好在有一些python的原子操作是线程安全的,比如读写单一基本变量(int, float, string)。
二、roscpp
c++版本的ROS1节点要复杂得多。roscpp建立了一个global callback queue回调队列,所有的subscriber和timer都将自己的回调add到这个callba