用SOCKET发送OpenCV的IplImage结构

来源:互联网 发布:js 骰子的转动效果 编辑:程序博客网 时间:2024/05/01 10:44

做一个C/S架构的Image Processing的程序,其中读写图片和图像处理的部分都是用OpenCV来实现的。本人原先没有怎么接触过SOCKET编程,网上也没有找到怎么发送一个类似IplImage结构这样文章,之后自己解决了,所以把CSDN的博客开了写一篇这个,自己记录一下,也方便像我一样的初学者。

对于一般的结构例如CvPoint,我们首先来看一下CvPoint的结构

typedef struct CvPoint{    int x;    int y;}CvPoint;

如果要发送一个CvPoint的话可以使用以下语句

CvPoint point(1, 1);send(s, (char*)&point, sizeof(CvPoint), 0);


这里使用的是TCP,阻塞模式的SOCKET(下同)。那么如果要在客户端接受这个CvPoint结构则可以这样:

CvPoint p;recv(s, (char*)&p, sizeof(CvPoint), 0);
实际上这种发送接受的结构类似于memcpy,发送的过程就是将内存中指向point所占空间的开始地址接连的sizeof(CvPoint)个字节的内容(实际上就是point的内容)拷贝到发送缓冲区中等待发送。

而接受则是将当前接收缓冲区中的内容连续拷贝sizeof(CvPoint)个字符到以&p为起点的内存空间中。比较简略的示意图如下所示

这时候实际上采用什么类型的指针来作为index都是无所谓的。例如&p为0x00cd4360,这时候声明一个int类型的指针指向p的话同样也是指向0x00cd4360.唯一不同的是如果你把它指向的地址元素以int形式取出来可能结果会很怪,另外int类型指针增加的话是跨越2个字节而不是1个。所以这里即使是用BYTE*代替char*都是可以的。

所以可能像我一样的初学者就觉得这样就可以如法炮制传送IplImage结构了,其实不是这样的。首先来分析一下IplImage的结构:

typedef struct _IplImage{    int  nSize;             /* sizeof(IplImage) */    int  ID;                /* version (=0)*/    int  nChannels;         /* Most of OpenCV functions support 1,2,3 or 4 channels */    int  alphaChannel;      /* Ignored by OpenCV */    int  depth;             /* Pixel depth in bits: IPL_DEPTH_8U, IPL_DEPTH_8S, IPL_DEPTH_16S,                               IPL_DEPTH_32S, IPL_DEPTH_32F and IPL_DEPTH_64F are supported.  */    char colorModel[4];     /* Ignored by OpenCV */    char channelSeq[4];     /* ditto */    int  dataOrder;         /* 0 - interleaved color channels, 1 - separate color channels.                               cvCreateImage can only create interleaved images */    int  origin;            /* 0 - top-left origin,                               1 - bottom-left origin (Windows bitmaps style).  */    int  align;             /* Alignment of image rows (4 or 8).                               OpenCV ignores it and uses widthStep instead.    */    int  width;             /* Image width in pixels.                           */    int  height;            /* Image height in pixels.                          */    struct _IplROI *roi;    /* Image ROI. If NULL, the whole image is selected. */    struct _IplImage *maskROI;      /* Must be NULL. */    void  *imageId;                 /* "           " */    struct _IplTileInfo *tileInfo;  /* "           " */    int  imageSize;         /* Image data size in bytes                               (==image->height*image->widthStep                               in case of interleaved data)*/    char *imageData;        /* Pointer to aligned image data.         */    int  widthStep;         /* Size of aligned image row in bytes.    */    int  BorderMode[4];     /* Ignored by OpenCV.                     */    int  BorderConst[4];    /* Ditto.                                 */    char *imageDataOrigin;  /* Pointer to very origin of image data                               (not necessarily aligned) -                               needed for correct deallocation */}IplImage;
其中我们通过分析IplImage的结构可以看出,实际上其中成员除了int,char数组之外还有char指针。所以我们可以看出来,IplImage是一个Header+Data式的结构。它的data项实际上是存在另外的地方,用一个imageData指示数据的首地址而已。所以分析到这里已经很明白了,只需要将IplImage的header和data分开传输过去就可以了。以下是简单实现的代码,其中hdr_data以及HDRLEN_DATA分别是数据包头和数据包头的长度,用来指示接下去要收多少个字节的data:

#define BUFF_SIZE 1024 * 4//any number you wants according to your OS#define HDRLEN_IPLIMAGE (sizeof(IplImage))IplImage* RcvIplImage(SOCKET s)//returns a pointer to the iplimage{IplImage header;IplImage* source;char *databuf;char *index;int ret;hdr_data datainfo;char buf[BUFF_SIZE+5];ret = ::recv(s, (char *)&header, HDRLEN_IPLIMAGE, 0);source = cvCreateImageHeader(cvSize(header.width, header.height), header.depth, header.nChannels);if (ret != HDRLEN_IPLIMAGE){return NULL;}databuf = new char[source->imageSize];source->imageData = databuf;source->imageDataOrigin = databuf;index = databuf;ret = ::recv(s, (char *)&datainfo, HDRLEN_DATA, 0);if (ret != HDRLEN_DATA){return NULL;}while(ret > 0 && datainfo.data_len != -1){receivedata(buf, datainfo.data_len, s);memcpy(index, buf, datainfo.data_len);index += datainfo.data_len;receivedata((char *)&datainfo, HDRLEN_DATA, s);}return source;}BOOL SendIplImage(IplImage* source, SOCKET s){int ret, count, datalen;char *index;hdr_data datainfo;index = source->imageData;count = 0;ret = ::send(s, (char *)source, HDRLEN_IPLIMAGE, 0);datalen = BUFF_SIZE;while (ret > 0){datainfo.data_len = datalen;ret = ::send(s, (char *)&datainfo, HDRLEN_DATA, 0);ret = ::send(s, (char *)index, datalen, 0);if (datalen < BUFF_SIZE)break;count += datalen;index += datalen;if (count > source->imageSize - BUFF_SIZE)datalen = source->imageSize - count;}datainfo.data_len = -1;ret = ::send(s, (char *)&datainfo, HDRLEN_DATA, 0);return TRUE;}BOOL receivedata(char* buf, int length, SOCKET s)//用来从接受缓冲区中接受length个字节{int ret, lefti;lefti = length;while (lefti > 0){ret = ::recv(s, (char *)buf, lefti, 0);buf += ret;lefti = lefti - ret;}return TRUE;}

以上就是发送和接受IplImage结构的代码。但是需要指出的是,对于RcvIplImage返回的IplImage *结构最好不要用CvReleaseImage来释放。因为这个函数是对应于使用CvCreateImage函数创建的图像的,直接使用可能因为CvFree的一些设计导致冲突。推荐使用以下代码来解决这个问题:

IplImage *oriImage = RcvIplImage(m_socket);delete[] (oriImage->imageData);cvReleaseImageHeader(&oriImage);
与此同时我也研究了一下怎么发送CvMat,这个比起IplImage来说有点麻烦,鉴于我已经很困了,所以下次再写

PS:给个最简单的hdr_data的定义

其实就是把Iplimage分成header和data,在接收端得到了包头之后把ipliamge的imageData指针指向接收到的data的的位置。
hdr_data是数据包头的内容,这个可以根据需要定义。比如说最简单地就用
struct hdr_data{
long length;
}

原创粉丝点击