c#调用C++DLL的办法

c#调用C++DLL的办法

LIBEXPORT_API

主要参考了两篇文章《C#中简单调用c/c++旧模块》 和《基于Visual C++6.0的DLL编程实现》,复杂的理论和概念就不说了,简略地写一个实现的全过程吧:

  1. 在Visual Studio 6.0中用C++语方创建dll文件。 新建工程时选择”Win32 Dynamic-link library ”,在DllMain.cpp文件中按如下方式声明函数:

#define LIBEXPORT_API extern “C” __declspec(dllexport)
下面展示一些 内联代码片

// dllmain.cpp : 定义 DLL 应用程序的入口点。
#include "pch.h"

#define LIBEXPORT_API extern "C" __declspec(dllexport)

extern "C" __declspec(dllexport) void ccc();

//用作导出函数
 void  aaa()
{
     MessageBoxA(NULL,"导出函数aaa被调用成功","信息",MB_OK);
}

 //要干的事情
 void  bbb()
{
     MessageBoxA(NULL, "导出函数bbb被调用成功", "信息", MB_OK);
}

 //擦屁股
 void  ccc()
{
     MessageBoxA(NULL, "导出函数ccc被调用成功", "信息", MB_OK);
}

BOOL APIENTRY DllMain( HMODULE hModule,
                       DWORD  ul_reason_for_call,
                       LPVOID lpReserved
                     )
{
    switch (ul_reason_for_call)
    {
    case DLL_PROCESS_ATTACH:
    case DLL_THREAD_ATTACH:
    case DLL_THREAD_DETACH:
    case DLL_PROCESS_DETACH:
        break;
    }
    return TRUE;
}

在DllMain.cpp中实现这个函数。 生成CJ.dll和CJ.lib。
2. 在Visual C# .net中引用dll文件 将CJ.dll和CJ.dll.lib拷贝到可执行文件目录下。

添加引用using System.Runtime.InteropServices;

按如下方式声明一个将要引用MyDll.dll中函数的类:

 [DllImport("CJ.dll", EntryPoint = "ccc", ExactSpelling = false, CallingConvention = CallingConvention.Cdecl)]
        static private extern void ccc();

背景

C#中调用C++的DLL
引自 http://www.cnblogs.com/fhuafeng/archive/2009/06/29/1513560.html
c++经过这么多年的发展已经积累了大量的动态连接库,如果能够在.net环境里应用这些函数库,
可以很大的提高整个应用的开发速度。
使用c++编程的人员肯定对指针不会感到陌生,由于c++中的函数接口好多都可能定义成位指针,
而c#中只有在声明为unsafe code中才能够使用指针。如果想让c++的DLL支持在C#中调用,
那么在C++接口的声明中需要使用下面的这种格式:
extern “C” __declspec(dllexport) void __stdcall popMessage(char* message)
{
MessageBox(NULL, message, “C message from C#!”, MB_OK);
}
并且在c#类声明中使用如下的导入编译好的DLL,例如:
[ DllImport( “test.dll”, CallingConvention=CallingConvention.Cdecl )]
public static extern void Message(string theMessage);
当然你可以从一个DLL中导入多个方法的声明,例如:
[ DllImport( “test.dll”, CallingConvention=CallingConvention.Cdecl )]
public static extern void Func1(string theMessage);
[ DllImport( “test.dll”, CallingConvention=CallingConvention.Cdecl )]
public static extern void Func2(string theMessage);
[ DllImport( “test.dll”, CallingConvention=CallingConvention.Cdecl )]
public static extern void Func3(string theMessage);

一、发生的背景

 在开发新项目中使用了新的语言开发C#和新的技术方案WEB Service,但是在新项目中,一些旧的模块需要继续使用,一般是采用C或C++或Delphi编写的,如何利用旧模块对于开发人员来说,有三种可用方法供选择:第一、将C或C++函数用C#彻底改写一遍,这样整个项目代码比较统一,维护也方便一些。但是尽管微软以及某些书籍说,C#和C++如何接近,但是改写起来还是很痛苦的事情,特别是C++里的指针和内存操作;第二、将C或C++函数封装成COM,在C#中调用COM比较方便,只是在封装时需要处理C或C++类型和COM类型之间的转换,也有一些麻烦,另外COM还需要注册,注册次数多了又可能导致混乱;第三、将C或C++函数封装成动态链接库,封装的过程简单,工作量不大。因此我决定采用加载动态链接库的方法实现,于是产生了在C#中如何调用自定义的动态链接库问题,我在网上搜索相关主题,发现一篇调用系统API的文章,但是没有说明如何解决此问题,在MSDN上也没有相关详细说明。基于此,我决定自己从简单出发,逐步试验,看看能否达到自己的目标。

(说明一点:我这里改写为什么很怕麻烦,我改写的代码是变长加密算法函数,代码有600多行,对算法本身不熟悉,算法中指针和内存操作太多,要想保证算法正确,最可行的方法就是少动代码,否则只要有一点点差错,就不能肯定算法与以前兼容)

二、技术实现

下面看看如何逐步实现动态库的加载,类型的匹配:

动态链接库函数导出的定义,这个不需要多说,大家参考下面宏定义即可:

#define LIBEXPORT_API extern “C” __declspec(dllexport)
第一步,我先从简单的调用出发,定义了一个简单的函数,该函数仅仅实现一个整数加法求和:

LIBEXPORT_API int mySum(int a,int b){ return a+b;}

C#定义导入定义:

public class RefComm

{

[DllImport(“LibEncrypt.dll”, EntryPoint=" mySum ",CharSet=CharSet.Auto,CallingConvention=CallingConvention.StdCall)] public static extern int mySum (int a,int b);

}

在C#中调用测试:

int iSum= RefComm. mySum(2,3);

运行查看结果iSum为5,调用正确。第一步试验完成,说明在C#中能够调用自定义的动态链接库函数。

第二步,我定义了字符串操作的函数(简单起见,还是采用前面的函数名),返回结果为字符串:

LIBEXPORT_API char *mySum(char *a,char *b){sprintf(b,”%s”,a) return a;}
C#定义导入定义:

public class RefComm

{

[DllImport(“LibEncrypt.dll”, EntryPoint=" mySum ",CharSet=CharSet.Auto,CallingConvention=CallingConvention.StdCall)] public static extern string mySum (string a, string b);

}

在C#中调用测试:

string strDest=””;

string strTmp= RefComm. mySum(“12345”, strDest);

运行查看结果strTmp为“12345”,但是strDest为空。

我修改动态链接库实现,返回结果为串b:

LIBEXPORT_API char *mySum(char *a,char *b){sprintf(b,”%s”,a) return b;}

修改C#导入定义,将串b修改为ref方式:

public class RefComm

{

[DllImport(“LibEncrypt.dll”, EntryPoint=" mySum ",CharSet=CharSet.Auto,CallingConvention=CallingConvention.StdCall)] public static extern string mySum (string a, ref string b);

}

在C#中再调用测试:

string strDest=””;

string strTmp= RefComm. mySum(“12345”, ref strDest);

运行查看结果strTmp和strDest均不对,含不可见字符。

再修改C#导入定义,将CharSet从Auto修改为Ansi:

public class RefComm

{

[DllImport(“LibEncrypt.dll”, EntryPoint=" mySum ",CharSet=CharSet.Ansi,CallingConvention=CallingConvention.StdCall)] public static extern string mySum (string a, string b);

}

在C#中再调用测试:

string strDest=””;

string strTmp= RefComm. mySum(“12345”, ref strDest);

运行查看结果strTmp为“12345”,但是串strDest没有赋值。第二步实现函数返回串,但是在函数出口参数中没能进行输出。
再次修改C#导入定义,将串b修改为引用(ref):
public class RefComm
{

[DllImport(“LibEncrypt.dll”, EntryPoint=" mySum ",CharSet=CharSet.Ansi,CallingConvention=CallingConvention.StdCall)] public static extern string mySum (string a, ref string b);

}

运行时调用失败,不能继续执行。

第三步,修改动态链接库实现,将b修改为双重指针:

LIBEXPORT_API char *mySum(char *a,char **b){sprintf((*b),”%s”,a) return *b;}

C#导入定义:

public class RefComm

{

[DllImport(“LibEncrypt.dll”, EntryPoint=" mySum ",CharSet=CharSet.Ansi,CallingConvention=CallingConvention.StdCall)] public static extern string mySum (string a, ref string b);

}

在C#中调用测试:

string strDest=””;

string strTmp= RefComm. mySum(“12345”, ref strDest);

运行查看结果strTmp和strDest均为“12345”,调用正确。第三步实现了函数出口参数正确输出结果。

第四步,修改动态链接库实现,实现整数参数的输出:
LIBEXPORT_API int mySum(int a,int b,int *c){ *c=a+b; return *c;}

C#导入的定义:

public class RefComm

{

[DllImport(“LibEncrypt.dll”, EntryPoint=" mySum ",CharSet=CharSet.Ansi,CallingConvention=CallingConvention.StdCall)]

public static extern int mySum (int a, int b,ref int c);

}

在C#中调用测试:

int c=0;
int iSum= RefComm. mySum(2,3, ref c);
运行查看结果iSum 和c均为5,调用正确。
经过以上几个步骤的试验,基本掌握了如何定义动态库函数以及如何在C#定义导入,有此基础,很快我实现了变长加密函数在C#中的调用,至此目标实现。

三、结论

在C#中,调用C++编写动态链接库函数,如果需要出口参数输出,则需要使用指针,对于字符串,则需要使用双重指针,对于C#的导入定义,则需要使用引用(ref)定义。

对于函数返回值,C#导入定义和C++动态库函数申明定义需要保持一致,否则会出现函数调用失败。

定义导入时,一定注意CharSet和CallingConvention参数,否则导致调用失败或结果异常。

运行时,动态链接库放在C#程序的目录下即可,我这里是一个C#的动态链接库,两个动态链接库就在同一个目录下运行。
然后你可以在你的c#类中调用上面声明的方法。

                                                  DLLImport 属性(MSDN)

准确地说,DllImport 属性具有下列行为:
  它只能放置在方法声明上。
  它具有单个定位参数:指定包含被导入方法的 dll 名称的 dllName 参数。
  它具有五个命名参数:
CallingConvention 参数指示入口点的调用约定。如果未指定 CallingConvention,则使用默认值
  CallingConvention.Winapi。
  CharSet 参数指示用在入口点中的字符集。如果未指定 CharSet,则使用默认值 CharSet.Auto。
  EntryPoint 参数给出 dll 中入口点的名称。如果未指定 EntryPoint,则使用方法本身的名称。
  ExactSpelling 参数指示 EntryPoint 是否必须与指示的入口点的拼写完全匹配。如果未指定
  ExactSpelling,则使用默认值 false。
  PreserveSig 参数指示方法的签名应当被保留还是被转换。当签名被转换时,它被转换为一个具有HRESULT 返回值和该返回值的一个名为 retval 的附加输出参数的签名。如果未指定 PreserveSig,则使用默认值 true。
  SetLastError 参数指示方法是否保留 Win32“上一错误”。如果未指定 SetLastError,则使用默认值false。
  它是一次性属性类。

此外,用 DllImport 属性修饰的方法必须具有 extern 修饰符。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值