OpenCV:IplImage和单字节ch…

对OpenCV稍有了解的同学都知道里边用于存储图像数据的IplImage,其中有两个属性非常值得关注,稍不留神就会导致错误(后附错例一则):一是width属性;二是widthStep属性。前者是表示图像的每行像素数,后者指表示存储一行像素需要的字节数。

      在OpenCV里边,widthStep必须是4的倍数,从而实现字节对齐,有利于提高运算速度。如果8U单通道图像宽度为3,那么widthStep是 4,加一个字节补齐。这个图像的一行需要4个字节,只使用前3个,最后一个空着。也就是一个宽3高3的图像的imageData数据大小为4*3=12字 节。

      需要注意的是,空着的那个像素并不是无效的,它仍然可以被操作,这就是导致错误的根源。

      错例:

      假如现在有一个char* data的指针指向一个15*15的灰度图像的数据起始地址,我们想把图像数据通过cvShowImage函数显示出来,比较直观的一种做法如下:

OpenCV:IplImage和单字节char*相互转换 - jinsuo2007 - 嵌入式工程师 ......
OpenCV:IplImage和单字节char*相互转换 - jinsuo2007 - 嵌入式工程师
OpenCV:IplImage和单字节char*相互转换 - jinsuo2007 - 嵌入式工程师IplImage
* image = cvCreateImage(cvSize(15, 15), 8, 1
);
OpenCV:IplImage和单字节char*相互转换 - jinsuo2007 - 嵌入式工程师memcpy(image
->imageData, data, 15*15
);
OpenCV:IplImage和单字节char*相互转换 - jinsuo2007 - 嵌入式工程师
OpenCV:IplImage和单字节char*相互转换 - jinsuo2007 - 嵌入式工程师cvNamedWindow(
"window"
);
OpenCV:IplImage和单字节char*相互转换 - jinsuo2007 - 嵌入式工程师cvShowImage("window", image);
OpenCV:IplImage和单字节char*相互转换 - jinsuo2007 - 嵌入式工程师
OpenCV:IplImage和单字节char*相互转换 - jinsuo2007 - 嵌入式工程师cvWaitKey();
OpenCV:IplImage和单字节char*相互转换 - jinsuo2007 - 嵌入式工程师
OpenCV:IplImage和单字节char*相互转换 - jinsuo2007 - 嵌入式工程师cvReleaseImage(
&
image);
OpenCV:IplImage和单字节char*相互转换 - jinsuo2007 - 嵌入式工程师cvDestroyWindow(
"window"
);
OpenCV:IplImage和单字节char*相互转换 - jinsuo2007 - 嵌入式工程师
OpenCV:IplImage和单字节char*相互转换 - jinsuo2007 - 嵌入式工程师......

      你会发现,显示的图像奇怪的往左下角歪过去了。当你看完这篇文章后希望不要再因为这个问题浪费你的时间了(shamed:这个问题郁闷了我整整一天)。其 实原因就在于,在cvCreateImage的时候,OpenCV为实现字节对齐,使得每行数据实际有16个字节(多出一个),在使用memcpy的过程 中,这些多出的字节就把对应的数据给“吃”了,因为这些数据在cvShowImage的时候并不会显示出来,这样,第二行就少一个字节,第三行少两个字 节,……,所以整个图像就偏向左下角了!

      知道这一点后可以将memcpy语句更改如下:

OpenCV:IplImage和单字节char*相互转换 - jinsuo2007 - 嵌入式工程师OpenCV:IplImage和单字节char*相互转换 - jinsuo2007 - 嵌入式工程师for(int i = 0; i<15; i++)... {
OpenCV:IplImage和单字节char*相互转换 - jinsuo2007 - 嵌入式工程师     memcpy(image
->imageData + image->widthStep*i, data + 15*i, 15
);
OpenCV:IplImage和单字节char*相互转换 - jinsuo2007 - 嵌入式工程师}

      这样,程序才能按我们的设想运行。

      上面曾指出了一个IplImage中imageData指针容易出错的问题, 这篇文章接着之前的叙述,提出IplImage和单字节char*之间相互转换的正确、简洁的方法:

      已知 IplImage* image 和 char* data

      从 IplImage 到 char* :

OpenCV:IplImage和单字节char*相互转换 - jinsuo2007 - 嵌入式工程师 data = image -> imageData // 对齐的图像数据

      或者

OpenCV:IplImage和单字节char*相互转换 - jinsuo2007 - 嵌入式工程师 data = image -> imageDataOrigin // 未对齐的原始图像数据

      从 char* 到 IplImage :

OpenCV:IplImage和单字节char*相互转换 - jinsuo2007 - 嵌入式工程师 image = cvCreateImageHeader(cvSize(width,height), depth, channels);
OpenCV:IplImage和单字节char*相互转换 - jinsuo2007 - 嵌入式工程师cvSetData(image, data, step);

      step指定IplImage图像每行占的字节数。需要注意是,在释放空间时不能直接使用cvReleaseImage,而需cvReleaseImageHeader,然后再delete data,这也是OpenCV里边“自己管理内存”的思想。

【转载】http://cau.anzhi.blog.163.com/blog/static/12577552008473242176/

Stkcd [股票代码] ShortName [股票简称] Accper [统计截止日期] Typrep [报表类型编码] Indcd [行业代码] Indnme [行业名称] Source [公告来源] F060101B [净利润现金净含量] F060101C [净利润现金净含量TTM] F060201B [营业收入现金含量] F060201C [营业收入现金含量TTM] F060301B [营业收入现金净含量] F060301C [营业收入现金净含量TTM] F060401B [营业利润现金净含量] F060401C [营业利润现金净含量TTM] F060901B [筹资活动债权人现金净流量] F060901C [筹资活动债权人现金净流量TTM] F061001B [筹资活动股东现金净流量] F061001C [筹资活动股东现金净流量TTM] F061201B [折旧摊销] F061201C [折旧摊销TTM] F061301B [公司现金流1] F061302B [公司现金流2] F061301C [公司现金流TTM1] F061302C [公司现金流TTM2] F061401B [股权现金流1] F061402B [股权现金流2] F061401C [股权现金流TTM1] F061402C [股权现金流TTM2] F061501B [公司自由现金流(原有)] F061601B [股权自由现金流(原有)] F061701B [全部现金回收率] F061801B [营运指数] F061901B [资本支出与折旧摊销比] F062001B [现金适合比率] F062101B [现金再投资比率] F062201B [现金满足投资比率] F062301B [股权自由现金流] F062401B [企业自由现金流] Indcd1 [行业代码1] Indnme1 [行业名称1] 季度数据,所有沪深北上市公司的 分别包含excel、dta数据文件格式及其说明,便于不同软件工具对数据的分析应用 数据来源:基于上市公司年报及公告数据整理,或相关证券交易所、各部委、省、市数据 数据范围:基于沪深北证上市公司 A股(主板、中小企业板、创业板、科创板等)数据整理计算
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值