学习Opencv(蝴蝶书/C++)——4.图形和大型数组类型(上)

2023-12-02 06:20

本文主要是介绍学习Opencv(蝴蝶书/C++)——4.图形和大型数组类型(上),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 1. cv::Mat类的成员变量
    • 1.1 flags
    • 1.2 cv::Mat::step
  • 2 存储方式,存储位置计算
    • 2.1 存储方式
    • 2.2 🌈存储位置计算
      • 2.2.1 基本计算公式
      • 2.2.1 step代码说明
      • 2.2.3 内存地址计算代码说明
  • 3 创建数据
    • 3.0 Mat的构成
      • 3.0.1 3.0版本之后的Mat
      • 3.0.2 cvMat
    • 3.1 构造函数
    • 3.2 🌈构造函数总结
  • 4 Mat访问(at<>, 迭代器和ptr)
    • 4.0 🌈数据类型(CV_16F)等
    • 4.1 🌈模板函数at<>()访问
      • 4.1.1 基本使用示例
      • 4.1.2 构造函数
    • 4.2 🌈数组迭代器访问
      • 4.2.1 示例(对比ptr效果,ptr一个小trick)
      • 4.2.2 构造函数MatConstIterator_和MatIterator_
      • 4.2.3 NAryMatIterator迭代器构造函数
    • 4.3 🌈ptr/指针访问(内存地址)
      • 4.3.1 示例
      • 4.3.2 构造函数
    • 4.4 三种访问方式速度比较
      • 4.4.1 基本说明
      • 4.4.2 代码

OpenCV库中的大部分函数,要么是cv::Mat类的成员,要么是使用cv::Mat类作为参数,要么就是把cv::Mat作为返回值,很少有函数和这三种都没有关系。

OpenCV中的cv::Mat就相当于Numpy中的array类型,是整个库最核心的概念,其它内容都是围绕这个概念建立起来的。

cv::Mat类表示N维稠密数组,cv::SparsesMat表示N维稀疏数组

1. cv::Mat类的成员变量

OpenCV2.x版本中的CvMat结构,与现在OpenCV3.x之后的还是有很大不同的。

现在的cv::Mat类的成员变量:

  • int flags,包括下面四种标志,
    • the magic signature,魔法标识
    • continuity flag,连续性标志
    • depth,深度
    • number of channels,通道数
  • int dims,表示矩阵的维度,>=2
  • int rows和int cols表示矩阵的高/行和宽/列,当维度>2时,这两个值无效,返回(-1,-1)
  • uchar * data,一个数据指针,指向数据真正存储的地方
  • MatStep step,定义矩阵的数据布局
  • MatSize size: Size(cols, rows) ,这是因为在Size的构造器里,rows和columns的参数顺序是相反的,如下:
    typedef Size2i cv::Size
    typedef Size_<int> cv::Size2i
    Size_ (_Tp _width, _Tp _height) size指定的先是宽cols,再是高rows
    

1.1 flags

以下回答来自chatGPT:

在 OpenCV 中,cv::Mat 类有一个特殊的成员变量 flags,它是一个整数,表示 cv::Mat 对象的属性和标志。

flags 的值是通过对不同的标志位进行按位或操作得到的,不同的标志位表示不同的属性和行为。以下是一些常用的 flags 标志和它们的作用:

  • cv::Mat::CONTINUOUS_FLAG:表示矩阵是连续的,即数据在内存中是按行存储的,没有间隔。
  • cv::Mat::DEPTH_MASK:表示矩阵的深度(数据类型),可以通过与 cv::Mat::DEPTH_MASK 进行按位与操作来提取深度值。
  • cv::Mat::CHANNELS_MASK:表示矩阵的通道数,可以通过与 cv::Mat::CHANNELS_MASK 进行按位与操作来提取通道数值。
  • cv::Mat::MAGIC_MASK:用于检测错误或非法的 cv::Mat 对象。
  • cv::Mat::TYPE_MASK:用于提取矩阵的类型信息,flags 中除去上述标志位以外的部分。

此外,还有一些特定的标志位,如 cv::Mat::CONTINUOUScv::Mat::CV_32S 等,用于表示特定的属性。

你可以使用 cv::Mat::flags 成员变量来获取 cv::Mat 对象的标志位,也可以使用 cv::Mat::type() 函数来获取矩阵的类型。例如:

cv::Mat image = cv::imread("image.jpg");
int flags = image.flags;
int type = image.type();

注意,通过 flags 获取的是一个整数,你可以通过按位与与按位右移等操作来提取具体的属性值。而通过 type() 函数获取的是一个表示类型的整数,通过与 CV_ 系列常量进行比较,可以获知矩阵的具体类型。

参考:

  • OpenCV学习之路(二)——Mat对象
  • 【OpenCV4】cv::Mat.isContinuous() 函数判断内存是否连续(c++
  • OpenCV】从Mat的flags中可以读到的信息,以及相关宏定义

1.2 cv::Mat::step

详见OpenCV:Mat中的step、elemSize和任意内存访问

在这里插入图片描述

2 存储方式,存储位置计算

2.1 存储方式

cv::Mat可以用来表示任意维度的数组,数组中数据的存储和n维光栅扫描顺序的类似。
如果是一维数组,其存储就是连续的;
如果是二维数组,数据被组织成行,然后按照行排列(以前数据结构里,数组排列的行优先)
如果是三维数组(plane),那就是每个plane先被逐行填满,然后plane再一个接一个。

例如:

有一维数组: [1,3,5,2,7]
其存储:①③⑤②⑦有二维数组(OpenCV里打印2d数组结果是这样的):
[1,3,5,2,7;
1,3,5,2,7;
1,3,5,2,7]
其存储:①③⑤②⑦|①③⑤②⑦|①③⑤②⑦
一个|代表一行三维数组。。。无法直接使用std::cout打印,可以理解为
其存储:
①③⑤②⑦|①③⑤②⑦|①③⑤②⑦||①③⑤②⑦|①③⑤②⑦|①③⑤②⑦||①③⑤②⑦|①③⑤②⑦|①③⑤②⑦
一个|代表一行,一个||表示一个plane

2.2 🌈存储位置计算

2.2.1 基本计算公式

1.1 成员变量中可以知道,对于一个矩阵 M M M,其布局是由数组M.step[]所定义的,因此索引为 [ i 0 , i 1 . . . , i M . d i m s − 1 ] [i_0,i_1...,i_{M.dims-1}] [i0,i1...,iM.dims1]的元素(其中 0 ≤ i k ≤ M . s i z e [ k ] 0\leq i_k \leq M.size[k] 0ikM.size[k])的地址可以通过下式计算:
a d d r ( M i 0 , i 1 . . . , i M . d i m s − 1 ) = M . d a t a + M . s t e p [ 0 ] ∗ i 0 + M . s t e p [ 1 ] ∗ i 1 + . . . . + M . s t e p [ M . d i m s − 1 ] ∗ i M . d i m s − 1 addr(M_{i_0,i_1...,i_{M.dims-1}})=M.data+M.step[0]*i_0+M.step[1]*i_1+....+M.step[M.dims-1]*i_{M.dims-1} addr(M

这篇关于学习Opencv(蝴蝶书/C++)——4.图形和大型数组类型(上)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/444281

相关文章

qtcreater配置opencv遇到的坑及实践记录

《qtcreater配置opencv遇到的坑及实践记录》我配置opencv不管是按照网上的教程还是deepseek发现都有些问题,下面是我的配置方法以及实践成功的心得,感兴趣的朋友跟随小编一起看看吧... 目录电脑环境下载环境变量配置qmake加入外部库测试配置我配置opencv不管是按照网上的教程还是de

C++ HTTP框架推荐(特点及优势)

《C++HTTP框架推荐(特点及优势)》:本文主要介绍C++HTTP框架推荐的相关资料,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1. Crow2. Drogon3. Pistache4. cpp-httplib5. Beast (Boos

JAVA数组中五种常见排序方法整理汇总

《JAVA数组中五种常见排序方法整理汇总》本文给大家分享五种常用的Java数组排序方法整理,每种方法结合示例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录前言:法一:Arrays.sort()法二:冒泡排序法三:选择排序法四:反转排序法五:直接插入排序前言:几种常用的Java数组排序

重新对Java的类加载器的学习方式

《重新对Java的类加载器的学习方式》:本文主要介绍重新对Java的类加载器的学习方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、介绍1.1、简介1.2、符号引用和直接引用1、符号引用2、直接引用3、符号转直接的过程2、加载流程3、类加载的分类3.1、显示

C++类和对象之初始化列表的使用方式

《C++类和对象之初始化列表的使用方式》:本文主要介绍C++类和对象之初始化列表的使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C++初始化列表详解:性能优化与正确实践什么是初始化列表?初始化列表的三大核心作用1. 性能优化:避免不必要的赋值操作2. 强

C++迭代器失效的避坑指南

《C++迭代器失效的避坑指南》在C++中,迭代器(iterator)是一种类似指针的对象,用于遍历STL容器(如vector、list、map等),迭代器失效是指在对容器进行某些操作后... 目录1. 什么是迭代器失效?2. 哪些操作会导致迭代器失效?2.1 vector 的插入操作(push_back,

python+OpenCV反投影图像的实现示例详解

《python+OpenCV反投影图像的实现示例详解》:本文主要介绍python+OpenCV反投影图像的实现示例详解,本文通过实例代码图文并茂的形式给大家介绍的非常详细,感兴趣的朋友一起看看吧... 目录一、前言二、什么是反投影图像三、反投影图像的概念四、反向投影的工作原理一、利用反向投影backproj

C#如何调用C++库

《C#如何调用C++库》:本文主要介绍C#如何调用C++库方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录方法一:使用P/Invoke1. 导出C++函数2. 定义P/Invoke签名3. 调用C++函数方法二:使用C++/CLI作为桥接1. 创建C++/CL

Java学习手册之Filter和Listener使用方法

《Java学习手册之Filter和Listener使用方法》:本文主要介绍Java学习手册之Filter和Listener使用方法的相关资料,Filter是一种拦截器,可以在请求到达Servl... 目录一、Filter(过滤器)1. Filter 的工作原理2. Filter 的配置与使用二、Listen

C++如何通过Qt反射机制实现数据类序列化

《C++如何通过Qt反射机制实现数据类序列化》在C++工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作,所以本文就来聊聊C++如何通过Qt反射机制实现数据类序列化吧... 目录设计预期设计思路代码实现使用方法在 C++ 工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作。由于数据类