1.服务端实现
功能包cpp02_service的src目录下,新建C++文件demo01_server.cpp,并编辑文件,输入如下内容:
/*
需求:编写服务端,接收客户端发送请求,提取其中两个整型数据,相加后将结果响应回客户端。
步骤:
1.包含头文件;
2.初始化 ROS2 客户端;
3.定义节点类;
3-1.创建服务端;
3-2.处理请求数据并响应结果。
4.调用spin函数,并传入节点对象指针;
5.释放资源。
*/
//包含头文件
#include"rclcpp/rclcpp.hpp"
#include"base_interfaces_demo/srv/add_ints.hpp"
//占位符
using std::placeholders::_1;
using std::placeholders::_2;
using base_interfaces_demo::srv::AddInts;
//创建节点类
class Ser_wqh:public rclcpp::Node{
public:
Ser_wqh()
:Node("ser_wqh") //初始化继承的Node节点名称
{
RCLCPP_INFO(this->get_logger(),"客户端正在创建!!!");
//创建服务端
service_1 = this->create_service<AddInts>("add_init",std::bind(&Ser_wqh::service_ss,this,_1,_2));
RCLCPP_INFO(this->get_logger(),"客户端创建成功!!!");
}
private:
//创建服务端对象
rclcpp::Service<AddInts>::SharedPtr service_1;
//处理请求数据结果并响应
void service_ss(const AddInts::Request::SharedPtr req,const AddInts::Response::SharedPtr res)
{
res->sum = req->num1+req->num2;
RCLCPP_INFO(this->get_logger(),"请求数据:(%d %d),响应结果:%d",req->num1,req->num2,res->sum);
}
};
int main(int argc,char *argv[])
{
//创建ROS节点
rclcpp::init(argc,argv);
//回调
auto server = std::make_shared<Ser_wqh>();
rclcpp::spin(server);
//释放
rclcpp::shutdown();
return 0;
}
2.客户端实现
功能包cpp02_service的src目录下,新建C++文件demo02_client.cpp,并编辑文件,输入如下内容:
/*
需求:编写客户端,发送两个整型变量作为请求数据,并处理响应结果。
步骤:
//前提要判断main中提交的参数是否正确
1.包含头文件;
2.初始化 ROS2 客户端;
3.定义节点类;
3-1.创建客户端;
3-2.等待服务连接;(对于通信,如果客户端连接不到服务器,不能发请求)
3-3.组织请求数据并发送;
4.创建对象指针调用其功能,并处理响应;
5.释放资源。
*/
// 1.包含头文件;
#include "rclcpp/rclcpp.hpp"
#include "base_interfaces_demo/srv/add_ints.hpp"
using base_interfaces_demo::srv::AddInts;
using namespace std::chrono_literals;
// 3.定义节点类;
class MinimalClient: public rclcpp::Node{
public:
MinimalClient():Node("minimal_client"){
// 3-1.创建客户端;
client = this->create_client<AddInts>("add_init"); //模板:服务接口 参数:服务话题名称 返回值:服务对象指针
RCLCPP_INFO(this->get_logger(),"客户端创建,等待连接服务端!");
}
// 3-2.等待服务连接;
bool connect_server(){ //,创建一个函数,返回bool值,连接成功返回true,否则返回false
while (!client->wait_for_service(1s)) //wait_for_service 在指定超时时间内连接服务器,如果连接上了,返回true,否则返回false
//这里循环连接 0取反 ,1秒连接服务器
{
if (!rclcpp::ok()) //判断ctrl+c按下,这里是bug rclcpp::ok()用于ros客户端时实状态的 这里的bug和多线程有关
{
RCLCPP_INFO(rclcpp::get_logger("rclcpp"),"强制退出!");
return false;
}
// RCLCPP_INFO(rclcpp::get_logger(),"服务连接中,请稍候...");
RCLCPP_INFO(this->get_logger(),"服务连接中,请稍候...");
}
return true;
}
// 3-3.组织请求数据并发送;
rclcpp::Client<AddInts>::FutureAndRequestId send_request(int32_t num1, int32_t num2){
auto request = std::make_shared<AddInts::Request>();
request->num1 = num1;
request->num2 = num2;
/*
*/
return client->async_send_request(request);
}
private:
rclcpp::Client<AddInts>::SharedPtr client;
};
int main(int argc, char ** argv)
{
/*
rclcpp这个库是不依赖
*/
if (argc != 3){ //argc代表此程序传入数据的个数
RCLCPP_INFO(rclcpp::get_logger("rclcpp"),"请提交两个整型数据!"); //先判断是否提交两个数据 直接调工作空间调用
return 1;
}
// 2.初始化 ROS2 客户端;
rclcpp::init(argc,argv);
// 4.创建对象指针并调用其功能;
auto client = std::make_shared<MinimalClient>(); //智能指针类型
bool flag = client->connect_server(); //接收bool返回值
if (!flag) //判断 为0取反进函数
{
RCLCPP_INFO(rclcpp::get_logger("rclcpp"),"服务连接失败!");
return 0;
}
auto response = client->send_request(atoi(argv[1]),atoi(argv[2])); //接收处理返回值 输入的数据传入argv1 和 argv2
// 处理响应
if (rclcpp::spin_until_future_complete(client,response) == rclcpp::FutureReturnCode::SUCCESS) //SUCCESS 成功 这个spin只spin一次 建议做成代码片段
{
RCLCPP_INFO(client->get_logger(),"请求正常处理");
RCLCPP_INFO(client->get_logger(),"响应结果:%d!", response.get()->sum);
} else {
RCLCPP_INFO(client->get_logger(),"请求异常");
}
// 5.释放资源。
rclcpp::shutdown();
}
3.编辑配置文件
1.packages.xml
在创建功能包时,所依赖的功能包已经自动配置了,配置内容如下:
<depend>rclcpp</depend>
<depend>base_interfaces_demo</depend>
2.CMakeLists.txt
CMakeLists.txt 中服务端和客户端程序核心配置如下:
find_package(ament_cmake REQUIRED)
find_package(rclcpp REQUIRED)
find_package(base_interfaces_demo REQUIRED)
add_executable(demo01_server src/demo01_server.cpp)
ament_target_dependencies(
demo01_server
"rclcpp"
"base_interfaces_demo"
)
add_executable(demo02_client src/demo02_client.cpp)
ament_target_dependencies(
demo02_client
"rclcpp"
"base_interfaces_demo"
)
install(TARGETS
demo01_server
demo02_client
DESTINATION lib/${PROJECT_NAME})
4.编译
终端中进入当前工作空间,编译功能包:
colcon build --packages-select cpp02_service
5.执行
当前工作空间下,启动两个终端,终端1执行服务端程序,终端2执行客户端程序。
终端1输入如下指令:
. install/setup.bash
ros2 run cpp02_service demo01_server
终端2输入如下指令:
. install/setup.bash
ros2 run cpp02_service demo02_client 100 200