在ROS中使用OpenCV进行简单的图像处理---代码实现篇

19 篇文章 1 订阅

原文链接

再上一篇blog中,笔者总结了ROS系统中使用OpenCV库的进行简单图像处理的原理、系统相关的设置和程序包的下载。在这篇博客中,笔者将从代码层面介绍如何实现在ROS系统中读取图片,并使用OpenCV进行图像处理,在返回结果。

实例:从ROS中读取图象,转换后将彩色图象变成灰度图象,并返回灰度图象,转换后在ROS下输出。

正文


1. 在ROS下创建工作空间


工作空间(work space)是ROS中非常重要的一个概念,可以把工作空间理解为一个大的工厂,里面的分成几个大的生产车间(package),每一个生产车间中会有若干个具有不同技能的工人(node)。当工厂运转时,每个车间中的工人(node)同时工作,他们通过话题(topic)进行信息沟通。各个大的车间之间也存在这互相依赖的关系,共同组成一个有机的整体。

因此在每次编写ROS下的程序时都应该先建立一个独立的工作空间,然后再不段的丰满它的功能。

方法如下:新建一个终端输入:

    mkdir -p cv_ws/src

    cd src

    catkin_init_workspace

    cd ..

    catkin_make


2. 在工作空间下创建程序包


创建好了工作空间,下一步需要创建程序包。在ROS中节点是实现某一个功能的可执行文件(工人),一个或者多个节点可以组成一个程序包。这样做便于代码的复用。程序包默认在工作空间中的src文件夹中创建,而node默认在程序包的src文件中创建(cpp文件)。

方法如下:在原来的终端下继续输入:

    cd src

    catkin_create_pkg robot_vision roscpp std_msgs cv_bridge image_transport sensor_msgs

    cd ..

    catkin_make
创建程序包的一般格式是catkin_create_pkg <name> <dependencies package>,在本程序中我们要用到除roscpp之外的三个程序包进行图片的转换工作

3. 创建.cpp源文件

在ubuntu系统下,没有类似vs2010那样的集成开发平台,编写程序只需要在文本中编写,命名时采用相应的后缀即可。在创建的程序包的src文件中创建一个文本文件,并命名为getImage.cpp。具体代码和注释如下:

#include<ros/ros.h> //ros标准库头文件
#include<iostream> //C++标准输入输出库
/*
  cv_bridge中包含CvBridge库
*/
#include<cv_bridge/cv_bridge.h> 
/*
  ROS图象类型的编码函数
*/
#include<sensor_msgs/image_encodings.h> 
/*
   image_transport 头文件用来在ROS系统中的话题上发布和订阅图象消息
*/
#include<image_transport/image_transport.h> 

//OpenCV2标准头文件
#include<opencv2/core/core.hpp>
#include<opencv2/highgui/highgui.hpp>
#include<opencv2/imgproc/imgproc.hpp>

static const std::string INPUT = "Input"; //定义输入窗口名称
static const std::string OUTPUT = "Output"; //定义输出窗口名称

//定义一个转换的类
class RGB_GRAY
{
private:
    ros::NodeHandle nh_; //定义ROS句柄
    image_transport::ImageTransport it_; //定义一个image_transport实例
    image_transport::Subscriber image_sub_; //定义ROS图象接收器
    //image_transport::Publisher image_pub_; //定义ROS图象发布器
public:
    RGB_GRAY()
      :it_(nh_) //构造函数
    {
        image_sub_ = it_.subscribe("camera/rgb/image_raw", 1, &RGB_GRAY::convert_callback, this); //定义图象接受器,订阅话题是“camera/rgb/image_raw”
       // image_pub_ = it_.publishe("", 1); //定义图象发布器
        //初始化输入输出窗口
        cv::namedWindow(INPUT);
        cv::namedWindow(OUTPUT);
    }
    ~RGB_GRAY() //析构函数
    {
         cv::destroyWindow(INPUT);
         cv::destroyWindow(OUTPUT);
    }
    /*
      这是一个ROS和OpenCV的格式转换回调函数,将图象格式从sensor_msgs/Image  --->  cv::Mat
    */
    void convert_callback(const sensor_msgs::ImageConstPtr& msg) 
    {
        cv_bridge::CvImagePtr cv_ptr; // 声明一个CvImage指针的实例

        try
        {
            cv_ptr =  cv_bridge::toCvCopy(msg, sensor_msgs::image_encodings::RGB8); //将ROS消息中的图象信息提取,生成新cv类型的图象,复制给CvImage指针
        }
        catch(cv_bridge::Exception& e)  //异常处理
        {
            ROS_ERROR("cv_bridge exception: %s", e.what());
            return;
        }

        image_process(cv_ptr->image); //得到了cv::Mat类型的图象,在CvImage指针的image中,将结果传送给处理函数   
    }
    /*
       这是图象处理的主要函数,一般会把图像处理的主要程序写在这个函数中。这里的例子只是一个彩色图象到灰度图象的转化
    */
    void image_process(cv::Mat img) 
    {
       cv::Mat img_out;
       cv::cvtColor(img, img_out, CV_RGB2GRAY);  //转换成灰度图象
       cv::imshow(INPUT, img);
       cv::imshow(OUTPUT, img_out);
       cv::waitKey(5);
    }
};

//主函数
int main(int argc, char** argv)
{
    ros::init(argc, argv, "RGB");
    RGB_GRAY obj;
    ros::spin();
}

4. 以下是对上面代码中核心的部分进行解释

1)头文件

#include<cv_bridge/cv_bridge.h>

头文件cv_bridge中包含了CvBridge类,而CvBridge中的API可以将ROS下的sensor_msgs/Image消息类型转化成cv::Mat

#include<sensor_msgs/image_encodings.h>

头文件sensor_msgs/Image是ROS下的图像的类型,这个头文件中包含对图像进行编码的函数。 

#include<image_transport/image_transport.h>

 

这个头文件中包含的是ImageTransport类,这个类提供ROS中图像的订阅和发布。

剩下的头文件就是如程序中的注释所示。

2)核心类的主要API

image_transport类:图像传输类,其功能和ROS中的Publisher和Subscriber差不多,但是不同的是这个类在发布和订阅图片消息的同时还附带这摄像头的信息。相比较之下,   在ROS中传送图片信息,使用image_transport类要高效的多。同时,这个类可以以不同的图像压缩形式传输图像例如JPG/PNG等等,也可以添加插件定义传输数据的压缩模式。
                                     API:
                                     image_transport::ImageTransport类:这个类成员可以定义某个话题的图像类型的发布器和订阅器(类似ros的句柄)。
                                     image_transport::Publisher类:这个类定义了image_transport中的发布器。
                                     image_transport::Subsciber类:这个类定义了image_transport中的订阅器。
cv_bridge类:这个类中提供的API主要功能是将图像从sensor_msgs/Image类型转化成cv::Mat类型。
                          API:
                          cv_bidge::CvImage类:cv_bridge中提供的数据结构,里面包括OpenCV中的cv::Mat类型的图像信息,图像编码方式,ROS头文件等等。要得到cv::Mat类型的 图像,只需要定义一个对象然后给出对象中的成员object.image即可,或者指针CvImagePtr,ptr->image。
                          cv_bridge::toCvCopy()方法:参数是ROS下的sensor_msgs/ImageConstPtr,和图像压缩类型(例如:sensor_msgs::image_encodings::RGB8)。其功能是实 现复制图像信息这样,得到副本,这样我们可以从副本的CvImage中提取cv::Mat类型的图像进行处理。核心!
                          cv_bridge::toCvShare()方法:参数同上,但是这个函数只是共享指针地址。ROS是不予许直接对图像格式的消息进行操作的。
 

5. 编译成可执行文件

在编写程序后,这个文本程序在编译成可执行文件之前是不能够运行的。首先在建立的robot_vision的程序包中的CMakeLists.txt文件中加入如下代码:

find_package(OpenCV REQUIRED)  
include_directories(${OpenCV_INCLUDE_DIRS}) 
add_executable(grayImage src/grayImage.cpp)  //将src中的文件添加成名字为grayImage的可执行文件
target_link_libraries(grayImage ${catkin_LIBRARIES} ${OpenCV_LIBS})  //将src中的文件添加成名字为grayImage的可执行文件
add_dependencies(grayImage robot_vision_generate_messages_cpp)  //给可执行文件添加依赖包

 返回到工作空间下编译。catkin_make 

find_package(OpenCV REQUIRED)  
include_directories(${OpenCV_INCLUDE_DIRS}) 
add_executable(grayImage src/grayImage.cpp)  
target_link_libraries(grayImage ${catkin_LIBRARIES} ${OpenCV_LIBS})  
add_dependencies(grayImage robot_vision_generate_messages_cpp)

6. 总结

以上就是在ROS下使用opencv所要做的所有工作。总体上来看就是因为opencv处理的图像格式和ROS系统实际提供的格式不匹配,故而有cv_bridge这个包作为沟通的桥梁。二本程序中使用image_transport代替传统的ROS中的Publisher和Subscriber是另一个突破。同时还要注意的是,ubuntu下的opencv2中的某些函数的形式和window中的不相同,使用的时候要注意。
 

  • 11
    点赞
  • 79
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
OpenCV(Open Source Computer Vision Library)是一款开源的计算机视觉库,专门为图像和视频处理任务设计,广泛应用于学术研究、工业应用以及个人项目。以下是关于OpenCV的详细介绍: 历史与发展 起源:OpenCV于1999年由英特尔公司发起,旨在促进计算机视觉技术的普及和商业化应用。该项目旨在创建一个易于使用、高效且跨平台的库,为开发者提供实现计算机视觉算法所需的基础工具。 社区与支持:随着时间的推移,OpenCV吸引了全球众多开发者和研究人员的参与,形成了活跃的社区。目前,OpenCV由非盈利组织OpenCV.org维护,并得到了全球开发者、研究机构以及企业的持续贡献和支持。 主要特点 跨平台:OpenCV支持多种操作系统,包括但不限于Windows、Linux、macOS、Android和iOS,确保代码能够在不同平台上无缝运行。 丰富的功能:库包含了数千个优化过的函数,涵盖了计算机视觉领域的诸多方面,如图像处理(滤波、形态学操作、色彩空间转换等)、特征检测与描述(如SIFT、SURF、ORB等)、物体识别与检测(如Haar级联分类器、HOG、DNN等)、视频分析、相机校正、立体视觉、机器学习(SVM、KNN、决策树等)、深度学习(基于TensorFlow、PyTorch后端的模型加载与部署)等。 高效性能:OpenCV代码经过高度优化,能够利用多核CPU、GPU以及特定硬件加速(如Intel IPP、OpenCL等),实现高速图像处理和实时计算机视觉应用。 多语言支持:尽管OpenCV主要使用C++编写,但它提供了丰富的API绑定,支持包括C、Python、Java、MATLAB、JavaScript等多种编程语言,方便不同领域的开发者使用。 开源与免费:OpenCV遵循BSD开源许可证发布,用户可以免费下载、使用、修改和分发库及其源代码,无需担心版权问题。 架构与核心模块 OpenCV的架构围绕核心模块构建,这些模块提供了不同层次的功能: Core:包含基本的数据结构(如cv::Mat用于图像存储和操作)、基本的图像和矩阵操作、数学函数、文件I/O等底层功能。 ImgProc:提供图像预处理、滤波、几何变换、形态学操作、直方图计算、轮廓发现与分析等图像处理功能。 HighGui:提供图形用户界面(GUI)支持,如图像和视频的显示、用户交互(如鼠标事件处理)以及简单的窗口管理。 VideoIO:负责视频的读写操作,支持多种视频格式和捕获设备。 Objdetect:包含预训练的对象检测模型(如Haar级联分类器用于人脸检测)。 Features2D:提供特征点检测(如SIFT、ORB)与描述符计算、特征匹配与对应关系估计等功能。 Calib3d:用于相机标定、立体视觉、多视图几何等问题。 ML:包含传统机器学习算法,如支持向量机(SVM)、K近邻(KNN)、决策树等。 DNN:深度神经网络模块,支持导入和运行预训练的深度学习模型,如卷积神经网络(CNN)。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值