服务端和客户端的关系实际上就是请求和响应的关系。我们在定义服务类型的时候实际上只是定义了请求时客户端发送的数据类型及个数,还有服务端响应时返回来的数据类型和个数。对C++语言而言、服务端和客户端同样是.cpp文件编译生成的可执行文件。
一、服务端:
1、cd到包目录下,然后在src文件夹下创建.cpp文件;
2、.cpp文件包括以下部分:
①包含(include)依赖库,如ros/ros.h、beginner_tutorials/AddTwoInts.h(已生成的服务类型头文件)等
②请求处理函数部分:
bool add(beginner_tutorials::AddTwoInts::Request &req, beginner_tutorials::AddTwoInts::Request &res)
{
客户端传来的数据存在req中,在此函数体处理请求,并将结果存在res中
}
③主函数部分: ros::init(argc,argv,"add_two_ints_server");
ros::NodeHandle n;
// 向ROS节点网络广播:当前节点提
供add_two_ints服务,这个服务通过调用add
函数完成,
// 需要这个服务的可以来请求当前节点
ros::ServiceServer service = n.advertiseService("add_two_ints",add);
ros::spin(); //自循环,等待请求
二、客户端:
1、cd到包目录下,然后在src文件夹下创建.cpp文件;
2、.cpp文件包括以下部分:
主函数: ros::init(argc,argv,"add_two_ints_client");
// 客户端需要通过命令行输入参数,这里是两个加数用空格隔开,
// 所以输入参数个数不等于3时属于非法输入
if(argc != 3){return;};
ros::NodeHandle n;
ros::ServiceClient client = n.serviceClient<beginner_tutorials::AddTwoInts>("add_two_ints");
beginner_tutorials::AddTwoInts srv; // 实例化需要请求的服务
srv.request.a = atoll(argv[1]); // 把要传递给服务端的参数存入服务srv中
srv.request.b = atoll(argv[2]);
if (client.call(srv))
{
如果服务端成功响应,返回值存在srv.response中,在此函数体处理结果
}
else return;
return;
三、CMakeLists.txt文件的设置:
1、添加可执行文件,即add_executable(add_two_ints_server src/talker.cpp);
add_executable(add_two_ints_client src/talker.cpp);
2、添加目标连接库,即target_link_libraries(add_two_ints_server ${catkin_LIBRARIES});
target_link_libraries(add_two_ints_client ${catkin_LIBRARIES});
3、添加对生成的消息文件的依赖项,即 add_dependencies(add_two_ints_server beginner_tutorials_generate_messages_cpp);
add_dependencies(add_two_ints_client beginner_tutorials_generate_messages_cpp);