当前位置: 首页 > backend >正文

OpenCV CUDA模块结构分析与形状描述符------在 GPU 上计算图像的原始矩(spatial moments)函数spatialMoments()

  • 操作系统:ubuntu22.04
  • OpenCV版本:OpenCV4.9
  • IDE:Visual Studio Code
  • 编程语言:C++11

算法描述

该函数用于在 GPU 上计算图像的原始矩(spatial moments)。这些矩可用于描述图像中物体的形状特征,如面积、质心等。

与 cv::cuda::moments(…) 不同的是,这个函数将结果写入一个 OutputArray 中,而不是返回结构体。因此它更适合需要手动处理矩数组的应用场景。

参数

参数名类型描述
srcInputArray输入图像,支持 CV_8U, CV_16U, 或 CV_32S 类型,单通道图像。如果 binaryImage 为 true,则应为二值图像(前景非零,背景为零)。
momentsOutputArray输出矩数组,是一个一维数组,长度由 order 决定(见下文),数据类型由 momentsType 指定。
binaryImageconst bool如果为 true,则假设输入图像是二值图像;否则按灰度图处理。默认为 false。
orderconst MomentsOrder要计算的矩的最大阶数,可选:
- FIRST_ORDER_MOMENTS:仅计算到一阶矩(共 4 个矩)
- SECOND_ORDER_MOMENTS:计算到二阶矩(共 9 个矩)
- THIRD_ORDER_MOMENTS:计算到三阶矩(共 16 个矩)
momentsTypeconst int矩的数据类型,通常为 CV_64F(双精度浮点数),也可以是 CV_32F。
streamStream&可选的 CUDA 流对象,用于异步执行。默认为同步执行(Stream::Null())。

注意:

为了获得最佳性能,请预先分配一个一维 GpuMat 用于存储矩(moments),其类型和大小必须足以容纳指定阶数下的所有图像矩。

例如:当 order == MomentsOrder::SECOND_ORDER_MOMENTS 且 momentsType == CV_32F 时,可以这样分配:

GpuMat momentsDevice(1, numMoments(MomentsOrder::SECOND_ORDER_MOMENTS), CV_32F);

下载矩数组后,可以在主机端使用 cv::Moments 构造函数轻松地计算中心矩(central moments)和归一化矩(normalized moments)。例如:

HostMem momentsHostMem(1, numMoments(MomentsOrder::SECOND_ORDER_MOMENTS), CV_32F);
momentsDevice.download(momentsHostMem, stream);
stream.waitForCompletion();
Mat momentsMat = momentsHostMem.createMatHeader();
cv::Moments cvMoments(momentsMat.at<float>(0), momentsMat.at<float>(1),momentsMat.at<float>(2), momentsMat.at<float>(3),momentsMat.at<float>(4), momentsMat.at<float>(5),momentsMat.at<float>(6), momentsMat.at<float>(7),momentsMat.at<float>(8), momentsMat.at<float>(9)
);

示例详见 OpenCV 贡献模块源码中的测试文件:
opencv_contrib_source_code/modules/cudaimgproc/test/test_moments.cpp 中的 CUDA_TEST_P(Moments, Async) 测试用例

返回值

无返回值。矩的结果通过 moments 参数输出。

代码示例

#include <opencv2/opencv.hpp>
#include <opencv2/cudaimgproc.hpp>
#include <iostream>int main()
{// 加载图像cv::Mat h_src = cv::imread("/media/dingxin/data/study/OpenCV/sources/images/Lenna.png", cv::IMREAD_GRAYSCALE);if (h_src.empty()){std::cerr << "无法加载图像!" << std::endl;return -1;}// 上传图像到 GPUcv::cuda::GpuMat d_src;d_src.upload(h_src);// 获取最大阶数下的矩数量int nMoments = cv::cuda::numMoments(cv::cuda::MomentsOrder::THIRD_ORDER_MOMENTS);// 创建输出矩阵cv::cuda::GpuMat d_moments;d_moments.create(1, nMoments, CV_64F);  // 存储所有矩值// 计算空间矩cv::cuda::spatialMoments(d_src, d_moments, false, cv::cuda::MomentsOrder::THIRD_ORDER_MOMENTS);// 下载结果cv::Mat h_moments;d_moments.download(h_moments);// 打印结果std::cout << "原始矩(spatial moments):" << std::endl;for (int i = 0; i < h_moments.cols; ++i){std::cout << "Moment[" << i << "] = " << h_moments.at<double>(i) << std::endl;}return 0;
}

运行结果

原始矩(spatial moments):
Moment[0] = 3.4715e+07
Moment[1] = 9.17605e+09
Moment[2] = 8.59234e+09
Moment[3] = 3.17604e+12
Moment[4] = 2.31525e+12
Moment[5] = 2.88234e+12
Moment[6] = 1.22333e+15
Moment[7] = 8.11918e+14
Moment[8] = 7.81631e+14
Moment[9] = 1.09583e+15
http://www.xdnf.cn/news/9768.html

相关文章:

  • Python自动化之selenium语句——元素点击、输入、清空和八大元素定位方法
  • 【保姆级教程】Windows部署LibreTV+cpolar实现远程影音库访问全步骤
  • PaddleOCR本地部署 (Python+Flask)
  • 【机器学习基础】机器学习入门核心算法:集成学习(Ensemble Learning)
  • 【.net core】SkiaSharp 如何在Linux上实现
  • ArkUI(方舟UI框架)介绍
  • MinVerse 3D触觉鼠标的技术原理与创新解析
  • MAZANOKE图像优化器本地部署与cpolar随时随地远程使用
  • 设计模式:观察者模式 - 实战
  • MATLAB中的table数据类型:高效数据管理的利器
  • OCC笔记:面、边的方向(TopAbs_Orientation)
  • Triton推理服务器部署YOLOv8(onnxruntime后端和TensorRT后端)
  • C++哈希
  • GitHub Copilot 使用手册与原理解析
  • 80x86CPU入栈与出栈操作
  • C++:lambda匿名函数
  • 【VSCode-Qt】Docker远程连接的项目UI文件在 VSCode 上无法预览
  • STM32F407VET6学习笔记8:UART5串口接收中断的Cubemx配置
  • Spring 面经
  • PnP(Perspective-n-Point)算法 | 用于求解已知n个3D点及其对应2D投影点的相机位姿
  • 5.LoadBalancer负载均衡服务调用
  • 【测试】Bug和用例
  • 视觉分析开发范例:Puppeteer截图+计算机视觉动态定位
  • 国内连接速度较快的常用 Yum 源及其具体配置方法
  • mysql一主多从 k8s部署实际案例
  • 【C++项目】:仿 muduo 库 One-Thread-One-Loop 式并发服务器
  • 什么是VR全景相机?如何选择VR全景相机?
  • http传输协议的加密
  • 黑马点评项目02——商户查询缓存(缓存穿透、缓存雪崩、缓存击穿)以及细节
  • 有关于常量的一节知识