ICode9

精准搜索请尝试: 精确搜索
首页 > 编程语言> 文章详细

C#调用C++ DLL类方法(转)--托管

2021-05-09 16:05:35  阅读:159  来源: 互联网

标签:MaskCLR C# Mask C++ DLL int maskValue inputFile


https://www.cnblogs.com/profession/p/5851749.html

C++的优势在于高效灵活,C#的优势在于简单易用,两者结合起来对项目开发来说是件好事,而且C++不容易反编译,也保障了代码的安全性,如果一些核心算法使用C#编写,那么保密就是一个问题。
C++生成的DLL一般只提供函数接口,不能直接用C#调用C++写的类,这样非常不方便,于是经过半天的反复测试,终于确定了一套可行的方案,下面就是整个流程的一个范例。

(1)编写C++类Mask,类前面的extern "C" class _declspec(dllexport)这么一串修饰符是必备的,不然无法导出类。
[cpp] view plain copy print?
//Mask类头文件mask.h
#pragma once

extern "C" class _declspec(dllexport) Mask
{
public:
Mask(char* inputFile,int* maskValue,int maskLength);
virtual void Run(void);
~Mask(void);

private:
char* _inputFile;
int* _maskValue;
int _maskLength;
};

(2)Mask类内部实现,这里使用char*和int*这两个有代表性的参数,C#向C++进行参数的传递是很难弄的一步。
[cpp] view plain copy print?
//Mask类实现文件mask.cpp
Mask::Mask(char* inputFile ,int* maskValue,int maskLength)
{
_inputFile=new char[strlen(inputFile)+1];
strcpy(_inputFile,inputFile);

_maskValue=new int[maskLength];
_maskLength=maskLength;
for(int i=0;i<maskLength;i++)
_maskValue[i]=maskValue[i];
}

void Mask::Run(void)
{
}

Mask::~Mask(void)
{
if (_inputFile)
{
delete [] _inputFile;
_inputFile=NULL;
}
if (_maskValue)
{
delete [] _maskValue;
_maskValue=NULL;
}
}

可以设置生成类型为DLL,然后将debug目录下的dll文件和lib文件找到,留待下一步使用。

(3)编写C++.NET类MaskCLR,C++.NET我一直认为就是个摆设,没想到也能派上用场,难得啊,也不完全是个废物。MaskCLR可以调用Mask类,注意函数的参数已经变成String ^和int*,现在是.NET环境,使用指针没那么方便了。
[cpp] view plain copy print?
//MaskCLR类头文件,用来包装Mask类MaskCLR.h
#pragma once
#include " mask.h" //这个就是上面的Mask头文件

public ref class MaskCLR
{
private:
char* _inputFile;
int* _maskValue;
int _maskLength;

public:
MaskCLR(String ^ inputFile,int* maskValue,int maskLength);
virtual void Run(void) override;
};

(4)MaskCLR类内部实现,首先要处理参数类型问题,将String ^类型转为char*。在MaskCLR:: Run函数内部调用Mask类,Mask是DLL导出的类。
[cpp] view plain copy print?
// MaskCLR类内部实现MaskCLR.cpp
#include "MaskCLR.h"

MaskCLR::MaskCLR(String ^ inputFile,int* maskValue,int maskLength)
{
_inputFile=GlobeFunction::StringToChar(inputFile);
_maskValue=maskValue;
_maskLength=maskLength;
}

void MaskCLR:: Run(void)
{
Mask mask(_inputFile, _maskValue,_maskLength);
mask.Run();
}

(5)将String ^类型转为char*,可以利用StringToHGlobalAnsi这个.NET自带的函数。
[csharp] view plain copy print?
char* GlobeFunction::StringToChar(String ^str)
{
return (char*)(Marshal::StringToHGlobalAnsi(str)).ToPointer();
}

(6)最终生成了两个DLL文件,一个是原始C++编写的,另一个是托管C++编写的,托管dll只是个外壳,最终还是调用原始dll,因此这两个dll要放在一起。下面终于进入C#环境了,首先将托管dll添加引用到C#工程中,C++.NET和C#可以直接互用。

(7)由于有个int*类型参数,在C#里指针属于不安全代码,因此使用unsafe关键字将涉及到指针的代码包括起来,在工程属性里设置允许使用不安全代码。定义int指针需要使用stackalloc关键字,创建一个int数组,对数组赋值后,将指针传递给类函数。
[csharp] view plain copy print?
//c#调用托管dll中的MaskCLR类form1.cs
unsafe
{
int* value = stackalloc int[1];
value[0] = 0;
MaskCLR mask = new MaskCLR("D:\\临时\\mask8.tif", value, 1);
mask.Run();
}

(8)至此大功告成,可以用C#很方便的调用C++写的类了.

标签:MaskCLR,C#,Mask,C++,DLL,int,maskValue,inputFile
来源: https://www.cnblogs.com/xihong2014/p/14748016.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有