因为需要使用c#调用c++编写的dll,且dll中涉及OpenCV函数调用,因此参数的传递可以采用emgucv提供的接口,而不用傻乎乎的将c#中的结构转为字节,再在c++中重新转为opencv的结构。
Dll中的C++函数如下,功能就是彩色图像yuv空间的直方图均衡:
- inthisto_enhance(IplImage* src, IplImage* dst)
- {
- cv::Mat img = cv::cvarrToMat(src);
- std::vector<cv::Mat> out;
- cv::Mat kk;
- cv::cvtColor(img, kk, CV_BGR2YUV);
- cv::split(kk, out);
-
-
- cv::Mat colorimt;
- cv::Ptr<cv::CLAHE> clahe =cv::createCLAHE();
- clahe->setClipLimit(3);
- clahe->setTilesGridSize(cv::Size(8,8));
-
- kk = out[0];
-
- clahe->apply(kk, colorimt);
- out.at(0) = colorimt;
- cv::Mat cc;
- cv::merge(out, cc);
-
-
- cv::cvtColor(cc, colorimt, CV_YUV2BGR);
- IplImage tmp = colorimt;
- *dst = tmp;
- return 0;
- }
这里讲dst作为参数传递,而不是返回值,目的就是在c#代码内申请对象空间,否则在c++函数内 new出来的对象即使采用Marshal.PtrToStructure转换为托管代码,依然会有莫名其妙的错误。
C#中的相关调用代码如下:
- IntPtr ptr = CvInvoke.cvCreateImage(CvInvoke.cvGetSize(frame),Emgu.CV.CvEnum.IPL_DEPTH.
- IPL_DEPTH_8U, 3);
- API.histo_enhance(frame.Ptr,ptr);
- MIplImage mlI =(MIplImage)Marshal.PtrToStructure(ptr, typeof(MIplImage));
- Image<Bgr, Byte> outframe = newImage<Bgr, Byte>(mlI.width, mlI.height, mlI.widthStep, mlI.imageData);
其中API定义如下:
- public class API
- {
- [DllImport("dll名称",EntryPoint = "histo_enhance", CallingConvention =CallingConvention.Cdecl)]
- public static extern inthisto_enhance(IntPtr src,IntPtr dst);
- }