c/c++中opencv实现腐蚀的示例代码

 更新时间:2025年06月03日 10:16:16   作者:whoarethenext  
图像腐蚀是形态学图像处理中的一种基本操作,OpenCV的cv::erode函数实现该功能,具有一定的参考价值,感兴趣的了解一下

图像腐蚀 (Erosion) 是形态学图像处理中的一种基本操作。它通常用于去除图像中的小型噪声、分离连接的物体以及细化二值图像中的物体。本文将详细介绍腐蚀的原理,并演示如何使用 C++ 和 OpenCV 库来实现图像腐蚀。

什么是图像腐蚀?

图像腐蚀可以被看作是用一个特定的结构元素(也称为核或掩模)“探测”或“扫描”输入图像的过程。对于二值图像,如果结构元素下的所有像素都为前景像素(通常为白色,值为1或255),则输出图像中结构元素锚点对应的像素才被置为前景像素,否则置为背景像素(通常为黑色,值为0)。对于灰度图像,输出像素的值是结构元素覆盖区域内所有像素的最小值。

简单来说,腐蚀操作会:

  • 缩小 亮色区域(前景)。
  • 扩大 暗色区域(背景)。
  • 消除小的、孤立的亮色噪点。

其效果就像是用结构元素“腐蚀”掉前景物体的边界。

结构元素 (Kernel)

结构元素是一个小的二值矩阵(或灰度矩阵),它定义了腐蚀操作的邻域范围和形状。结构元素的中心点通常被称为“锚点”。OpenCV 提供了 cv::getStructuringElement() 函数来方便地创建常见的结构元素。

常用的结构元素形状包括:

  • cv::MORPH_RECT: 矩形
  • cv::MORPH_ELLIPSE: 椭圆形
  • cv::MORPH_CROSS: 十字形
// 创建一个 5x5 大小的矩形结构元素
cv::Mat kernel_rect = cv::getStructuringElement(cv::MORPH_RECT, cv::Size(5, 5));

// 创建一个 5x5 大小的椭圆形结构元素
cv::Mat kernel_ellipse = cv::getStructuringElement(cv::MORPH_ELLIPSE, cv::Size(5, 5));

// 创建一个 5x5 大小的十字形结构元素
cv::Mat kernel_cross = cv::getStructuringElement(cv::MORPH_CROSS, cv::Size(5, 5));

结构元素的大小和形状对腐蚀结果有显著影响。

OpenCV 中的 cv::erode() 函数

OpenCV 提供了 cv::erode() 函数来实现图像腐蚀。

函数原型:

void cv::erode(
    cv::InputArray src,         // 输入图像
    cv::OutputArray dst,        // 输出图像
    cv::InputArray kernel,      // 用于腐蚀的结构元素
    cv::Point anchor = cv::Point(-1,-1), // 结构元素内锚点的位置。默认值 (-1,-1) 表示锚点在核中心。
    int iterations = 1,         // 腐蚀操作迭代次数
    int borderType = cv::BORDER_CONSTANT, // 像素外推方法
    const cv::Scalar& borderValue = cv::morphologyDefaultBorderValue() //边界不变时的边界值
);

参数说明:

  • src: 输入图像,可以是任意通道数的图像,但深度应为 CV_8UCV_16UCV_16SCV_32F 或 CV_64F
  • dst: 输出图像,与 src 具有相同的大小和类型。
  • kernel: 结构元素。如果传入一个空的 cv::Mat(),则默认使用一个 3x3 的矩形核。
  • anchor: 锚点位置。默认值 cv::Point(-1,-1) 表示锚点在结构元素的中心。
  • iterations: 腐蚀操作执行的次数。多次迭代等同于使用一个由原始核通过自身膨胀(相应次数)得到的更大的核。
  • borderType: 用于推断图像外部像素的边界模式。
  • borderValue: 当 borderType 为 cv::BORDER_CONSTANT 时使用的边界值。

C++ 代码示例

下面是一个完整的 C++ 示例,演示了如何加载图像并对其进行腐蚀操作:

#include <opencv2/opencv.hpp>
#include <iostream>

int main(int argc, char** argv) {
    // 1. 加载图像
    // 将 "your_image_path.png" 替换为你的图像路径
    cv::Mat srcImage = cv::imread("your_image_path.png", cv::IMREAD_COLOR);

    // 检查图像是否成功加载
    if (srcImage.empty()) {
        std::cerr << "错误: 无法加载图像!" << std::endl;
        return -1;
    }

    // 2. 创建腐蚀后的输出图像
    cv::Mat erodedImage;

    // 3. 定义结构元素 (核)
    // 尝试不同的形状和大小
    // int erosion_size = 3; // 核大小,可以调整
    // cv::Mat element = cv::getStructuringElement(cv::MORPH_RECT,
    //                                           cv::Size(2 * erosion_size + 1, 2 * erosion_size + 1),
    //                                           cv::Point(erosion_size, erosion_size));

    // 使用一个简单的 5x5 矩形核
    cv::Mat element = cv::getStructuringElement(cv::MORPH_RECT, cv::Size(5, 5));
    // 你也可以尝试:
    // cv::Mat element = cv::getStructuringElement(cv::MORPH_ELLIPSE, cv::Size(5, 5));
    // cv::Mat element = cv::getStructuringElement(cv::MORPH_CROSS, cv::Size(5, 5));

    // 如果不指定核,OpenCV 会默认使用一个 3x3 的矩形核
    // cv::Mat element; // 使用默认 3x3 核

    // 4. 执行腐蚀操作
    // 可以指定迭代次数,例如 iterations = 2
    int iterations = 1;
    cv::erode(srcImage, erodedImage, element, cv::Point(-1,-1), iterations);

    // 5. 显示原始图像和腐蚀后的图像
    cv::imshow("原始图像", srcImage);
    cv::imshow("腐蚀后的图像", erodedImage);

    // 6. 等待用户按键,然后关闭窗口
    cv::waitKey(0);
    cv::destroyAllWindows();

    return 0;
}

编译和运行说明 (以 g++ 为例):

g++ your_source_file.cpp -o erode_example $(pkg-config --cflags --libs opencv4)
./erode_example

(请确保你已安装 OpenCV 并配置好 pkg-config,如果使用 OpenCV 3,将 opencv4 替换为 opencv)

代码解析

  • #include <opencv2/opencv.hpp>: 包含了 OpenCV 的主要头文件。
  • cv::imread("your_image_path.png", cv::IMREAD_COLOR): 加载指定路径的彩色图像。你需要将 "your_image_path.png" 替换为实际的图像文件路径。
  • srcImage.empty(): 检查图像是否成功加载。
  • cv::getStructuringElement(cv::MORPH_RECT, cv::Size(5, 5)): 创建一个 5x5 大小的矩形结构元素。你可以修改形状 (MORPH_ELLIPSEMORPH_CROSS) 和大小 (cv::Size) 来观察不同的效果。
  • cv::erode(srcImage, erodedImage, element, cv::Point(-1,-1), iterations): 对 srcImage 应用腐蚀操作,使用 element 作为结构元素,迭代 iterations 次,并将结果存储在 erodedImage 中。cv::Point(-1,-1) 表示锚点在核的中心。
  • cv::imshow(): 显示原始图像和处理后的图像。
  • cv::waitKey(0): 等待用户按下任意键。
  • cv::destroyAllWindows(): 关闭所有 OpenCV 创建的窗口。

腐蚀的效果和应用场景

  • 去除小的独立噪声点: 对于二值图像中的“椒盐噪声”中的“盐”点(孤立的白色像素点),腐蚀操作非常有效。
  • 细化物体轮廓: 使二值图像中前景物体的尺寸变小,线条变细。
  • 分离物体: 如果两个物体之间有细微的连接,腐蚀操作可以断开这些连接,从而将它们分离。
  • 获取物体的核心区域: 多次腐蚀可以逐步剥离物体的外层像素,最终得到物体的核心部分。
  • 在其他更复杂形态学操作(如开运算、闭运算)中作为基础步骤。

可调整的参数及其影响

  • 结构元素的大小:
    • 较小的结构元素产生的腐蚀效果较弱,对图像的改变较小。
    • 较大的结构元素产生的腐蚀效果较强,会更显著地缩小或消除物体。
  • 结构元素的形状:
    • 矩形核对水平和垂直方向的腐蚀程度相同。
    • 线型核(例如,一个水平或垂直的细长矩形)会主要在与核垂直的方向上腐蚀物体。例如,水平线核会使物体在垂直方向上变细。
    • 十字形核和椭圆形核根据其形状特性进行腐蚀。
  • 迭代次数 (iterations):
    • 增加迭代次数会增强腐蚀效果,类似于使用一个更大的结构元素(但不完全相同,多次小核迭代与一次大核迭代在边界处理上可能有细微差别)。

总结

图像腐蚀是形态学图像处理中一个强大且基础的工具。通过选择合适的结构元素大小、形状以及迭代次数,可以有效地修改图像中物体的形状和大小,以达到去噪、分割和特征提取等目的。OpenCV 的 cv::erode() 函数为我们提供了一个简单易用的接口来实现这一功能。鼓励读者多多尝试不同的参数组合,观察它们对图像处理结果的具体影响。

到此这篇关于c/c++中opencv实现腐蚀的示例代码的文章就介绍到这了,更多相关opencv 腐蚀内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • C++中回调函数及函数指针的实例详解

    C++中回调函数及函数指针的实例详解

    这篇文章主要介绍了C++中回调函数及函数指针的实例详解的相关资料,希望通过本文能帮助到大家,让大家理解掌握这部分内容,需要的朋友可以参考下
    2017-10-10
  • vscode 配置 C/C++ 编译环境的详细图文教程

    vscode 配置 C/C++ 编译环境的详细图文教程

    这篇文章主要介绍了vscode 配置 C/C++ 编译环境的详细教程,本文通过图文并茂的形式给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-05-05
  • C++多重继承及多态性原理实例详解

    C++多重继承及多态性原理实例详解

    这篇文章主要介绍了C++多重继承及多态性原理实例详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-07-07
  • C++ vector的常见用法超详细讲解

    C++ vector的常见用法超详细讲解

    这篇文章主要介绍了C++ vector的常见用法,包括C++中vector容器的定义、初始化方法、访问元素、常用函数及其时间复杂度,通过代码介绍的非常详细,需要的朋友可以参考下
    2025-04-04
  • windows下用c++获取本机ip地址的三种方法

    windows下用c++获取本机ip地址的三种方法

    工作过程中遇到一个需求,需要获取本机ip地址,同时获取本机网络连接情况,即网线是否连接,经过多番搜索,本文给大家介绍了3种方案,通过代码示例介绍的非常详细,需要的朋友可以参考下
    2023-11-11
  • OpenCV霍夫变换(Hough Transform)直线检测详解

    OpenCV霍夫变换(Hough Transform)直线检测详解

    这篇文章主要为大家详细介绍了OpenCV霍夫变换直线检测的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-12-12
  • C++容器适配器的概念与示例

    C++容器适配器的概念与示例

    C++ STL(标准模板库)是一套功能强大的 C++ 模板类,提供了通用的模板类和函数,这些模板类和函数可以实现多种流行和常用的算法和数据结构,如向量、链表、队列、栈,今天我们来探究一下stl容器适配器的使用吧
    2023-01-01
  • C++空类默认函数详细解析

    C++空类默认函数详细解析

    如果你只是声明一个空类,不做任何事情的话,编译器会自动为你生成一个默认构造函数、一个拷贝默认构造函数、一个默认拷贝赋值操作符和一个默认析构函数
    2013-10-10
  • 输出1000以内的素数的算法(实例代码)

    输出1000以内的素数的算法(实例代码)

    本篇文章是对输出1000以内的素数的算法进行了详细的分析介绍,需要的朋友参考下
    2013-05-05
  • C++之内存分区的实现示例

    C++之内存分区的实现示例

    本文主要介绍了C++之内存分区的实现示例,主要包含了4个区域,分为代码区,全局区,栈区和堆区,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起学习学习吧
    2024-03-03

最新评论