opencv 表示图像的IplImage

article/2025/10/5 11:14:03

一、IplImage
大部分内容摘抄自http://blog.csdn.net/xiaowei_cqu/article/details/7557063
OpenCV资料:http://blog.csdn.net/to_utopia/article/details/4856171
IplImage是表示一个图像的结构体,因为之前的OpenCV是用C语言编写的,所以IplImage提供的接口是C语言接口。

typedef struct _IplImage  {  int  nSize;         /* IplImage大小 */  int  ID;            /* 版本 (=0)*/  int  nChannels;     /* 大多数OPENCV函数支持1,2,3 或 4 个通道 */  int  alphaChannel;  /* 被OpenCV忽略 */  int  depth;         /* 像素的位深度: IPL_DEPTH_8U, IPL_DEPTH_8S, IPL_DEPTH_16U, IPL_DEPTH_16S, IPL_DEPTH_32S, IPL_DEPTH_32F and       IPL_DEPTH_64F 可支持 */  char colorModel[4]; /* 被OpenCV忽略 */  char channelSeq[4]; /* 同上 */  int  dataOrder;     /* 0 - 交叉存取颜色通道, 1 - 分开的颜色通道. cvCreateImage只能创建交叉存取图像 */  int  origin;        /* 0 - 顶—左结构, 1 - 底—左结构 (Windows bitmaps 风格) */  int  align;         /* 图像行排列 (4 or 8). OpenCV 忽略它,使用 widthStep 代替 */  int  width;         /* 图像宽像素数 */  int  height;        /* 图像高像素数*/  struct _IplROI *roi;/* 图像感兴趣区域. 当该值非空只对该区域进行处理 */  struct _IplImage *maskROI; /* 在 OpenCV中必须置NULL */  void  *imageId;     /* 同上*/  struct _IplTileInfo *tileInfo; /*同上*/  int  imageSize;     /* 图像数据大小(在交叉存取格式下imageSize=image->height*image->widthStep),单位字节*/  char *imageData;  /* 指向排列的图像数据 */  int  widthStep;   /* 排列的图像行大小,以字节为单位 */  int  BorderMode[4]; /* 边际结束模式, 被OpenCV忽略 */  int  BorderConst[4]; /* 同上 */  char *imageDataOrigin; /* 指针指向一个不同的图像数据结构(不是必须排列的),是为了纠正图像内存分配准备的 */  }  IplImage;  

创建方法:
1,如果是从新创造一个Iplimage,则用IplImage* cvCreateImage( CvSize size, int depth, int channels ),它创建头并分配数据。
注:当不再使用这个新图像时,要调用void cvReleaseImage( IplImage** image )将它的头和图像数据释放!
2,如果有图像数据没有为图像头分配存储空间(即,没有为IplImage*指针分配动态存储空间),则先调用IplImage* cvCreateImageHeader( CvSize size, int depth, int channels )创建图像头,再调用void cvSetData( CvArr* arr, void* data, int step )指定图像数据,可以理解为将这个新图像的数据指针指向了一个已存在的图像数据上,不存在图像数据存储空间的分配操作。
注:当不再使用这个新图像时,要调用void cvReleaseImageHeader( IplImage** image )将它的图像头释放!
3,如果有图像数据也有图像头(用于IplImage为静态分配存储空间的情况),则先调用IplImage* cvInitImageHeader( CvSize size, int depth, int channels )更改图像头,再调用void cvSetData( CvArr* arr, void* data, int step )指定图像数据。
注:因为这个新图像使用的是其它图像的数据和已有的图像头,所以不能使用cvReleaseImage将它的头和图像数据释放,也不能使用cvReleaseData将它的图像数据释放!
4,如果从已有的一个图像创建,则用IplImage* cvCloneImage( const IplImage* image ),它制作图像的完整拷贝包括头、ROI和数据。
注:当不再使用这个新图像时,要调用void cvReleaseImage( IplImage** image )将它的头和图像数据释放!

直接访问:
对我们来说比较重要的两个元素是:char *imageData以及widthStep。imageData存放图像像素数据,而widStep类似CvMat中的step,表示以字节为单位的行数据长度。
一个m*n的单通道字节型图像,其imageData排列如下:
这里写图片描述
如果我们要遍历图像中的元素,只需:

IplImage* img=cvCreateImage(cvSize(640,480),IPL_DEPTH_8U,1);  
uchar* tmp;  
for(int i=0;i<img->height;i++)  for(int j=0;j<img->width;j++)  *tmp=((uchar *)(img->imageData + i*img->widthStep))[j];  

多通道浮点型

IplImage* img=cvCreateImage(cvSize(640,480),IPL_DEPTH_32F,3);
((float *)(img->imageData + i*img->widthStep))[j*img->nChannels + 0]=111; // B
((float *)(img->imageData + i*img->widthStep))[j*img->nChannels + 1]=112; // G
((float *)(img->imageData + i*img->widthStep))[j*img->nChannels + 2]=113; // R

这种直接访问的方法速度快,但容易出错,我们可以通过定义指针来访问。即:

IplImage* img=cvCreateImage(cvSize(640,480),IPL_DEPTH_8U,1);  
ucha* data=(uchar *)img->imageData;  
int step = img->widthStep/sizeof(uchar);  
uchar* tmp;  
for(int i=0;i<img->height;i++)  for(int j=0;j<img->width;j++)  *tmp=data[i*step+j];  

而多通道(三通道)字节图像中,imageData排列如下:
这里写图片描述

其中(Bi,Bj)(Gi,Gj)(Ri,Rj)表示图像(i,j)处BGR分量的值。使用指针的遍历方法如下:

IplImage* img=cvCreateImage(cvSize(640,480),IPL_DEPTH_8U,1);  
IplImage* img=cvCreateImage(cvSize(640,480),IPL_DEPTH_8U,3);  
uchar* data=(uchar *)img->imageData;  
int step = img->widthStep/sizeof(uchar);  
int channels = img->nChannels;  
uchar *b,*g,*r;  
for(int i=0;i<img->height;i++)  for(int j=0;j<img->width;j++){  *b=data[i*step+j*chanels+0];  *g=data[i*step+j*chanels+1];  *r=data[i*step+j*chanels+2];  } 

使用cvGet2D()函数访问图像某位置的像素值:
cvGet*D系列函数可以用来返回特定位置的数组元素(一般使用cvGet2D),原型如下:

CvScalar cvGet1D( const CvArr* arr, int idx0 );  
CvScalar cvGet2D( const CvArr* arr, int idx0, int idx1 );  
CvScalar cvGet3D( const CvArr* arr, int idx0, int idx1, int idx2 );  
CvScalar cvGetND( const CvArr* arr, int* idx );  

idx0,idx1,idx2分别用来指示元素数组下标,即cvGet2D返回(idx0,idx1)处元素的值。
因此,单通道图像像素访问方式如下:

IplImage* img=cvCreateImage(cvSize(640,480),IPL_DEPTH_8U,1);  
double tmp;  
for(int i=0;i<img->height;i++)  for(int j=0;j<img->width;j++)  tmp=cvGet2D(img,i,j).val[0];  

多通道字节型/浮点型图像:

IplImage* img=cvCreateImage(cvSize(640,480),IPL_DEPTH_32F,3);  
double tmpb,tmpg,bmpr;  
for(int i=0;i<img->height;i++)  for(int j=0;j<img->width;j++){  tmpb=cvGet2D(img,i,j).val[0];  tmpg=cvGet2D(img,i,j).val[1];  tmpr=cvGet2D(img,i,j).val[2];  } ```
如果是修改元素的值,可用cvSet*D(一般是cvSet2D)函数:这种方法对于任何图像的访问方式是一样的,比较简单,但效率较低,不推荐使用。

void cvSet1D( CvArr* arr, int idx0, CvScalar value );
void cvSet2D( CvArr* arr, int idx0, int idx1, CvScalar value );
void cvSet3D( CvArr* arr, int idx0, int idx1, int idx2, CvScalar value );
void cvSetND( CvArr* arr, int* idx, CvScalar value );

也可通过以下方式访问像素并修改(nRowIndex,nColumIndex)

uchar* ptr;
ptr=(uchar*)(img.imageData + nRowIndex* img.widthStep);
if (ptr[nColumIndex] != 255)
{
ptr[nColumIndex] = 255;//(nRowIndex,nColumIndex)的像素值修改为255
}
“`


http://chatgpt.dhexx.cn/article/h6XgHuvs.shtml

相关文章

IplImage结构体

一、IplImage的一些重要成员&#xff1a; 1、origin&#xff1a;图像原点的定义。0,则图片的左上角是原点&#xff1b;1&#xff0c;则左下角是原点。                                    IplImage* imgcvL…

CvMat、Mat、IplImage之间的转换详解及实例

IplImage&#xff1a; 在OpenCV中IplImage是表示一个图像的结构体&#xff0c;也是从OpenCV1.0到目前最为重要的一个结构&#xff1b; 在之前的图像表示用IplImage&#xff0c;而且之前的OpenCV是用C语言编写的&#xff0c;提供的接口也是C语言接口&#xff1b; Mat&#xff…

VMware 虚拟机 Ubuntu 系统执行 ifconfig 命令 eth0没有IP地址(intet addr、Bcast、Mask) 解决:UP BROADCAST MULTICAST 问题

VMware 虚拟机 ifconfig没有net_addr地址、Bcast、Mask的解决方法 最新更新时间&#xff1a;2018-11-9 02:43:55 使用时间长的虚拟机&#xff0c;会莫名其妙的连接不上网 在终端中&#xff0c;使用ifconfig命令查看Ubuntu系统的IP地址&#xff0c;发现没有分配IP地址。 eth0&a…

FISCO BCOS简介

FISCO BCOS是由国内企业主导研发、对外开源、安全可控的企业级金融联盟链底层平台&#xff0c;由金链盟开源工作组协作打造&#xff0c;并于2017年正式对外开源。 社区以开源链接多方&#xff0c;截止2020年5月&#xff0c;汇聚了超1000家企业及机构、逾万名社区成员参与共建共…

使用MPI实现broadcast、scatter、gather操作

使用MPI实现broadcast、scatter、gather操作 1.MPI_Bcast:广播消息 MPI_Bcast用于将一个进程的buffer中的数据广播到其他进程的相同buffer变量中。 #include "stdio.h" #include "mpi.h" #include "stdlib.h" #define N 10 int main(i…

用MPI进行分布式内存编程(二)MPI_allreduce MPI_scatt MPI_bcast.....

通过上一篇中&#xff0c;知道了基本的MPI编写并行程序&#xff0c;最后的例子中&#xff0c;让使用0号进程做全局的求和的所有工作&#xff0c;而其他的进程却都不工作&#xff0c;这种方式也许是某种特定情况下的方案&#xff0c;但明显不是最好的方案。举个例子&#xff0c;…

MPI编程(4)—集合通信MPI_Bcast、MPI_Gather、MPI_Scatter、MPI_Reduce

1. MPI_Bcast:广播消息 MPI_Bcast用于将一个进程的buffer中的数据广播到其他进程的相同buffer变量中 代码示例如下: int rank, data[10];MPI_Init(0, 0);MPI_Comm_rank(MPI_COMM_WORLD, &rank);if (rank == 0) {for (int i = 0; i < 10; ++i){data[i] = i + 1;}} //…

MPI_Bcast与MPI_Comm_split配合,实现行广播或列广播

15个进程&#xff0c;3行5列&#xff0c;结合MPI_Comm_split函数&#xff0c;MPI_Bcast实现行广播和列广播。 Bcast广播时&#xff0c;如果通讯域是split之后的&#xff0c;则会按照color相同的进行广播&#xff0c;即color相同的为一组&#xff0c;每个组内编号为root的向组内…

VMware 虚拟机 linux执行 ifconfig 命令 eth0没有IP地址(intet addr、Bcast、Mask) UP BROADCAST MULTICAST 问题

VMware 虚拟机 ifconfig没有net_addr地址、Bcast、Mask的解决方法&#xff1a; 在vmvare中&#xff0c;使用ifconfig命令查看linux系统的IP地址&#xff0c;发现没有分配IP地址。 eth0&#xff1a;网络接口 link encap : 网络类型 HWaddr : 网卡物理地址 Inet addr &…

CABAC

CABAC CABAC&#xff08;上下文自适应的二进制算术编码&#xff09;基于算术编码&#xff0c;在HEVC中&#xff0c;除了参数集、SEI和slice头部之外&#xff0c;其余的所有数据都使用CABAC来进行熵编码。1.4.1 原理 主要包括三个步骤&#xff1a; 二进制化&#xff1b; 上下…

【Redis】 - Redis 6.0 新特性之客户端缓存

Redis 6.0 新特性之客户端缓存 1. 为什么需要客户端缓存1.1 低延迟和大规模提供数据服务1.2 其他 cache 层 2. Redis 中的客户端缓存2.1 什么样的数据集应该被客户端缓存2.2 客户端缓存的两个主要优点 3. 缓存的数据一致性问题4. Redis 客户端缓存的实现原理4.1 普通模式4.1.1 …

MCAST是什么?(播协议通信程序)

看海康isapi.pdf文档看到的&#xff0c;不懂什么意思 播协议通信程序&#xff0c;用于测试在局域网或者在三层交换机架构下进行通信测试 Multicast Protocol communications procedures, for testing in the local area network or switches in the three-tier framework for …

一、Broadcast简介

Android广播&#xff08;Broadcast&#xff09; 一、Broadcast简介 Broadcast是android中的四大组件之一&#xff0c;是在组件之间传播数据&#xff08;Intent&#xff09;的一种机制。广播的发送者和接收者事先是不需要知道对方的存在的。这样带来的好处便是&#xff0c;系统的…

BCSP

BCSP 包的类型 有4种类型&#xff0c;Sync,Sync-Resp,Conf,Conf-Resp Sync: {0xda,0xdc,0xed,0xed} Sync-Resp: {0xac,0xaf,0xef,0xee} Conf: {0xad,0xef,0xac,0xed} Conf-Resp: {0xde,0xad,0xd0,0xd0} 上面2图式BCSP的行为和状…

MPI MPI_Bcast (广播)用法详解

函数范式 int MPI_Bcast(void * data_p;int count;MPI_Datatype datatype;int source_proc;MPI_Comm comm; );功能描述&#xff1a; 一个广播发生的时候&#xff0c;一个进程会把同样一份数据传递给一个 communicator 里的所有其他进程。根节点调用 MPI_Bcast 函数的时候&…

关于inet addr(网络地址)、bcast(广播地址)、mask(子网掩码)的学习

int addr:网络地址。IP地址和子网掩码进行相与运算&#xff0c;将运算结果中的网络地址不变&#xff0c;主机地址变为0&#xff0c;即主机号全0是网络地址。示例: 如果是192的C段地址&#xff0c;那么网络地址就是&#xff1a;192.168.1.0&#xff0c;地址掩码是&#xff1a;25…

错误码415

POST请求报错&#xff1a; “status”: 415, “error”: “Unsupported Media Type”, “message”: "Content type app charlse 错误码 415表示不支持内容类型。 错误原因 没有正确设置内容类型标头。 正确做法 检查 content-type

发送HTTP请求返回415状态码的解决办法

1.问题出现&#xff1a; 在用POST方式携带token访问一个API接口时&#xff0c;发现返回状态码为415&#xff0c;错误信息为“Unsupported Media Type” 2.分析原因&#xff1a; 415错误的解释是说&#xff0c;服务器无法处理请求附带的媒体格式&#xff0c;不明白什么意思&am…

http post 415错误

前面在spring boot项目中&#xff0c;进行前后端数据交互时突然遇到了415错误。一时间有点懵&#xff0c;在网上找了一些资料&#xff0c;总算是知道为什么了&#xff0c;不多说&#xff0c;直接上图。 这是正常的form提交的数据。 debug之后能够服务器能够正常的接收数据。 当…

http405错误解决

问题描述&#xff1a;在使用vue框架和spring boot进行前后端分离开发时&#xff0c;出现405错误。 解决思考&#xff1a; 分析&#xff1a;我们知道&#xff0c;一般4开头的错误基本都是前端的问题&#xff0c;或者是浏览器的问题。而且我之前使用了swagger对后端接口进行了测…