ROS學習筆記(一):自己動手寫一個ROS程序


創建一個工作區(workspace)

工作區可以作為一個獨立的項目進行編譯,存放ROS程序的源文件、編譯文件和執行文件。建立工作區的方法如下:

 

[plain]  view plain  copy
 
  1. $ mkdir -p ~/catkin_ws/src  
  2. $ cd ~/catkin_ws/src  
  3. $ catkin_init_workspace  

雖然這時候工作區是空的,但是我們依然可以進行編譯:

 

 

[plain]  view plain  copy
 
  1. $ cd ~/catkin_ws/  
  2. $ catkin_make  

 

這時候,會在當前文件夾下生成devel,build這兩個子文件夾,在devel文件夾下能看到幾個setup.*sh文件。

接下來把工作區在bash中注冊

 

[plain]  view plain  copy
 
  1. $ source devel/setup.bash  

要驗證是否已經在bash中注冊可以使用如下命令:

 

 

[plain]  view plain  copy
 
  1. $ echo $ROS_PACKAGE_PATH  
  2. /home/youruser/catkin_ws/src:/opt/ros/indigo/share:/opt/ros/indigo/stacks  

如果能看到自己工作區的文件路徑就說明已經成功了。

 

創建一個ROS工程包(Package)

在一個工作區內,可能會包含多個ROS工程包。而最基本ROS工程包中會包括CmakeLists.txt和Package.xml這兩個文件,其中Package.xml中主要包含本項目信息和各種依賴(depends),而CmakeLists.txt中包含了如何編譯和安裝代碼的信息。

 

首先切換到工作區:

 

[plain]  view plain  copy
 
  1. $ cd ~/catkin_ws/src  

現在可以使用catkin_create_pkg命令去創建一個叫beginner_tutorials的包,這個包依靠std_msgs、roscpp、rospy。

 

 

[plain]  view plain  copy
 
  1. $ catkin_create_pkg beginner_tutorials std_msgs rospy roscpp  

接下來在工作區編譯這個工程包。

 

 

[plain]  view plain  copy
 
  1. $ cd ~/catkin_ws  
  2. $ catkin_make  

 

一個簡單的發布(Publisher)、訂閱(Subscriber)程序

 

寫一個發布(Publisher)節點

節點(node)是連接到ROS網絡中可執行的基本單元。我們在這創建一個發布者---“talker”節點,這個節點持續對外發布消息。

首先我們要把目錄切換到我們的beginner_tutorials工程包中

 

[plain]  view plain  copy
 
  1. $ cd ~/catkin_ws/src/beginner_tutorials  

因為我們已經編譯過這個工程包了,所以會在beginner_tutorials文件夾下看到CmakeList.txt、package.xml文件和include、src這兩個目錄。接下來進入src子目錄

 

 

[plain]  view plain  copy
 
  1. $ cd src  

在src目錄中創建一個talker.cpp文件,里面的內容如下:

 

 

[cpp]  view plain  copy
 
  1. #include "ros/ros.h"  
  2. #include "std_msgs/String.h"  
  3.   
  4. #include <sstream>  
  5. int main(int argc, char **argv)  
  6. {  
  7.   /** 
  8.    * The ros::init() function needs to see argc and argv so that it can perform 
  9.    * any ROS arguments and name remapping that were provided at the command line. For programmatic 
  10.    * remappings you can use a different version of init() which takes remappings 
  11.    * directly, but for most command-line programs, passing argc and argv is the easiest 
  12.    * way to do it.  The third argument to init() is the name of the node. 
  13.    * 
  14.    * You must call one of the versions of ros::init() before using any other 
  15.    * part of the ROS system. 
  16.    */  
  17.   ros::init(argc, argv, "talker");  
  18.   
  19.   /** 
  20.    * NodeHandle is the main access point to communications with the ROS system. 
  21.    * The first NodeHandle constructed will fully initialize this node, and the last 
  22.    * NodeHandle destructed will close down the node. 
  23.    */  
  24.   ros::NodeHandle n;  
  25.   
  26.   /** 
  27.    * The advertise() function is how you tell ROS that you want to 
  28.    * publish on a given topic name. This invokes a call to the ROS 
  29.    * master node, which keeps a registry of who is publishing and who 
  30.    * is subscribing. After this advertise() call is made, the master 
  31.    * node will notify anyone who is trying to subscribe to this topic name, 
  32.    * and they will in turn negotiate a peer-to-peer connection with this 
  33.    * node.  advertise() returns a Publisher object which allows you to 
  34.    * publish messages on that topic through a call to publish().  Once 
  35.    * all copies of the returned Publisher object are destroyed, the topic 
  36.    * will be automatically unadvertised. 
  37.    * 
  38.    * The second parameter to advertise() is the size of the message queue 
  39.    * used for publishing messages.  If messages are published more quickly 
  40.    * than we can send them, the number here specifies how many messages to 
  41.    * buffer up before throwing some away. 
  42.    */  
  43.   ros::Publisher chatter_pub = n.advertise<std_msgs::String>("chatter", 1000);  
  44.   
  45.   ros::Rate loop_rate(10);  
  46.   
  47.   /** 
  48.    * A count of how many messages we have sent. This is used to create 
  49.    * a unique string for each message. 
  50.    */  
  51.   int count = 0;  
  52.   while (ros::ok())  
  53.   {  
  54.     /** 
  55.      * This is a message object. You stuff it with data, and then publish it. 
  56.      */  
  57.     std_msgs::String msg;  
  58.   
  59.     std::stringstream ss;  
  60.     ss << "hello world " << count;  
  61.     msg.data = ss.str();  
  62.   
  63.     ROS_INFO("%s", msg.data.c_str());  
  64.   
  65.     /** 
  66.      * The publish() function is how you send messages. The parameter 
  67.      * is the message object. The type of this object must agree with the type 
  68.      * given as a template parameter to the advertise<>() call, as was done 
  69.      * in the constructor above. 
  70.      */  
  71.     chatter_pub.publish(msg);  
  72.   
  73.     ros::spinOnce();  
  74.   
  75.     loop_rate.sleep();  
  76.     ++count;  
  77.   }  
  78.   
  79.   
  80.   return 0;  
  81. }  


寫一個訂閱Subscriber節點

還是在src目錄下,創建一個listener.cpp文件。內容如下:

 

[cpp]  view plain  copy
 
  1. #include "ros/ros.h"  
  2. #include "std_msgs/String.h"  
  3.   
  4. /** 
  5.  * This tutorial demonstrates simple receipt of messages over the ROS system. 
  6.  */  
  7. void chatterCallback(const std_msgs::String::ConstPtr& msg)  
  8. {  
  9.   ROS_INFO("I heard: [%s]", msg->data.c_str());  
  10. }  
  11.   
  12. int main(int argc, char **argv)  
  13. {  
  14.   /** 
  15.    * The ros::init() function needs to see argc and argv so that it can perform 
  16.    * any ROS arguments and name remapping that were provided at the command line. For programmatic 
  17.    * remappings you can use a different version of init() which takes remappings 
  18.    * directly, but for most command-line programs, passing argc and argv is the easiest 
  19.    * way to do it.  The third argument to init() is the name of the node. 
  20.    * 
  21.    * You must call one of the versions of ros::init() before using any other 
  22.    * part of the ROS system. 
  23.    */  
  24.   ros::init(argc, argv, "listener");  
  25.   
  26.   /** 
  27.    * NodeHandle is the main access point to communications with the ROS system. 
  28.    * The first NodeHandle constructed will fully initialize this node, and the last 
  29.    * NodeHandle destructed will close down the node. 
  30.    */  
  31.   ros::NodeHandle n;  
  32.   
  33.   /** 
  34.    * The subscribe() call is how you tell ROS that you want to receive messages 
  35.    * on a given topic.  This invokes a call to the ROS 
  36.    * master node, which keeps a registry of who is publishing and who 
  37.    * is subscribing.  Messages are passed to a callback function, here 
  38.    * called chatterCallback.  subscribe() returns a Subscriber object that you 
  39.    * must hold on to until you want to unsubscribe.  When all copies of the Subscriber 
  40.    * object go out of scope, this callback will automatically be unsubscribed from 
  41.    * this topic. 
  42.    * 
  43.    * The second parameter to the subscribe() function is the size of the message 
  44.    * queue.  If messages are arriving faster than they are being processed, this 
  45.    * is the number of messages that will be buffered up before beginning to throw 
  46.    * away the oldest ones. 
  47.    */  
  48.   ros::Subscriber sub = n.subscribe("chatter", 1000, chatterCallback);  
  49.   
  50.   /** 
  51.    * ros::spin() will enter a loop, pumping callbacks.  With this version, all 
  52.    * callbacks will be called from within this thread (the main one).  ros::spin() 
  53.    * will exit when Ctrl-C is pressed, or the node is shutdown by the master. 
  54.    */  
  55.   ros::spin();  
  56.   
  57.   return 0;  
  58. }  


編譯創建的節點
在編譯我們創建的節點之前,我們還需要編輯Cmakelist.txt文件(注意:是beginner_tutorials項目包下的CMakelist文件),告訴編輯器我們需要編輯什么文件,需要什么依賴。

 

 

[plain]  view plain  copy
 
  1. $ gedit CMakeLists.txt  

在文件末尾添加如下語句:

 

 

[plain]  view plain  copy
 
  1. include_directories(include ${catkin_INCLUDE_DIRS})  
  2.   
  3. add_executable(talker src/talker.cpp)  
  4. target_link_libraries(talker ${catkin_LIBRARIES})  
  5. add_dependencies(talker beginner_tutorials_generate_messages_cpp)  
  6.   
  7. add_executable(listener src/listener.cpp)  
  8. target_link_libraries(listener ${catkin_LIBRARIES})  
  9. add_dependencies(listener beginner_tutorials_generate_messages_cpp)  


將目錄切換到工作區目錄,並執行catkin_make運行命令:

 

 

[plain]  view plain  copy
 
  1. $ cd ~/catkin_ws  
  2. $ catkin_make  


不出意外的話,會出現如下界面:

 


至此,程序已經創建完成,而接下來我們要檢查一下我們創建的程序是否正確。

測試程序的正確性

首先,我們得要啟動ROS核心程序roscore。
[plain]  view plain  copy
 
  1. $ roscore  

在使用我們的程序之前,需要先把程序注冊
[plain]  view plain  copy
 
  1. $ cd ~/catkin_ws  
  2. $ source ./devel/setup.bash  

運行talker節點:
[plain]  view plain  copy
 
  1. $ rosrun beginner_tutorials talker   

這時候會看到如下信息:
[plain]  view plain  copy
 
  1. [INFO] [WallTime: 1314931831.774057] hello world 1314931831.77  
  2. [INFO] [WallTime: 1314931832.775497] hello world 1314931832.77  
  3. [INFO] [WallTime: 1314931833.778937] hello world 1314931833.78  
  4. [INFO] [WallTime: 1314931834.782059] hello world 1314931834.78  
  5. [INFO] [WallTime: 1314931835.784853] hello world 1314931835.78  
  6. [INFO] [WallTime: 1314931836.788106] hello world 1314931836.79  
這就表示發布(Publisher)節點已經正確的運行了。

接下來運行listener節點:
[plain]  view plain  copy
 
  1. $ rosrun beginner_tutorials listener  

這時候會看到如下信息:
[plain]  view plain  copy
 
  1. [INFO] [WallTime: 1314931969.258941] /listener_17657_1314931968795I heard hello world 1314931969.26  
  2. [INFO] [WallTime: 1314931970.262246] /listener_17657_1314931968795I heard hello world 1314931970.26  
  3. [INFO] [WallTime: 1314931971.266348] /listener_17657_1314931968795I heard hello world 1314931971.26  
  4. [INFO] [WallTime: 1314931972.270429] /listener_17657_1314931968795I heard hello world 1314931972.27  
  5. [INFO] [WallTime: 1314931973.274382] /listener_17657_1314931968795I heard hello world 1314931973.27  
  6. [INFO] [WallTime: 1314931974.277694] /listener_17657_1314931968795I heard hello world 1314931974.28  
  7. [INFO] [WallTime: 1314931975.283708] /listener_17657_1314931968795I heard hello world 1314931975.28  
這說明訂閱節點(listener)已經成功的接收到了發布節點(talker)發布的信息。至此,整個程序結束!


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM