C#调用非托管动态库中的函数方法

C#如何调用一个非托管动态库中的函数呢,比如用VC6写的动态库,总之C#调用动态库的过程是比Java调用DLL动态库方便快捷多了,下面举例说明这个过程。

1、创建一个非托管动态库

代码如下:

复制代码 代码如下:

//这一句是声明动态库输出一个可供外不调用的函数原型. 
   extern   "C"  __declspec(dllexport)  int  add( int ,  int );

int  add( int  a, int  b)  
   { 
        //实现这个函数returna+b; 
   }

注意上面代码,一定要加上 extern"C" ,不能生成的动态库中的导出函数名就不会是add,而是像 ?add@@YAHHH@Z 样子,后面只是通过函数名 add 来定位函数入口就会出问题。

保存成C或者CPP文件都可以,接下来就用命令 cl (这个命令VC6提供) 来编译生成一个动态库,命令如下:

复制代码 代码如下:

C:\>cl /LD MyLib.cpp
Microsoft (R) 32-bit C/C++ Optimizing Compiler Version 12.00.8804 for 80x86
Copyright (C) Microsoft Corp 1984-1998. All rights reserved.

MyLib.cpp
Microsoft (R) Incremental Linker Version 6.00.8447
Copyright (C) Microsoft Corp 1992-1998. All rights reserved.

/out:MyLib.dll
/dll
/implib:MyLib.lib
MyLib.obj
Creating library MyLib.lib and object MyLib.exp

可以看到在C盘根目录下生成了你要的动态库 MyLib.Dll ,还伴随着生成了MyLib.lib、MyLib.obj、MyLib.exp文件,上面命令cl的参数/LD就是生成动态库文件

2、编写C#程序调用动态库

复制代码 代码如下:

using  System; 
   using  System.Runtime.InteropServices; //这是用到DllImport时候要引入的包 
   public   class  InvokeDll{

[DllImport( "MyLib.dll" , CharSet=CharSet.Auto)] 
       staticexternint add( int  a, int  b); //声明外部的标准动态库, 跟Win32API是一样.

public   static   void  Main() 
       { 
         Console.WriteLine(add(10,30)); 
       } 
   }

保存为InvokeDll.cs文件, 与MyLib.dll置于同一目录, 编译该文件.

复制代码 代码如下:

C:\>csc InvokeDll.cs
Microsoft (R) Visual C# .NET 编译器版本 7.10.3052.4
用于 Microsoft (R) .NET Framework 版本 1.1.4322
版权所有 (C) Microsoft Corporation 2001-2002。保留所有权利。
将生成Invokedll.exe, 可以执行该文件.

C:\>InvokeDll
40

我们看到C#调用了非托管动态库的函数 add 。执行前保证 MyLib.dll 在能够被 InvokeDll 程序加载到的路径上。

回过头来,如果我们在MyLib.cpp中没有加上 extern"C" 在,那么C中通过函数名 add 定位不到导出方法(因为函数名在动态库中已经变了),执行invokeDll时就会出现如下错误。

复制代码 代码如下:

C:\>InvokeDll

未处理的异常: System.EntryPointNotFoundException: 无法在 DLL MyLib.dll 中找到名为 add 的入口点。
at InvokeDll.add(Int32 a, Int32 b)
at InvokeDll.Main()

对于没有加上 extern "C" 的函数原型生成的动态库,我们就得用别的方式来调用了,具体怎么做,我现在还不知道。下面还有一个问题,上面的例子只是演示了动态库中函数非常简单的情况,如果函数传递的参数是指针,或者更复杂的数据类型,又如何操作呢?以后会深究的。

[注:]本文参考着网上一篇文章:C-Sharp调用标准动态库 ,但是直接照着原文的操作就是会无法定位 add 的入口点的错误,所以略有修改。

时间: 2015-02-15

C#使用ILGenerator动态生成函数的简单代码

游戏服务器里面总是有一大堆的配置文件需要读取, 而且这些配置文件的读取: * 要不然做成弱类型的, 就是一堆字符串或者数字, 不能看出来错误(需要重新检测一次) * 要不然做成强类型的, 每种类型都需要自己Parse一下 我个人比较喜欢后者, 因为前者LoadConfig的代码简单, 但是写逻辑的时候代码不简单. class Config1 : public IConfig { public void Fill(EntryLine& line); int32_t param1; string p

史上最简单的MyBatis动态SQL入门示例代码

假如有如下的关于书籍基本信息的表: DROP DATABASE IF EXISTS `books`; CREATE DATABASE `books`; USE books; DROP TABLE IF EXISTS `book`; CREATE TABLE `book` ( `id` int(11) NOT NULL AUTO_INCREMENT, `name` varchar(128) DEFAULT NULL, `author` varchar(64) DEFAULT NULL, `pres

php动态生成函数示例

以下就是php动态生成函数示例,示例中用到了eval函数,我感觉如果服务器上允许用户运行这样的函数是非常危险的 复制代码 代码如下: <?$a['a']=1;$a['b']=1;$a['c']=1;$str="function a(){global \$a;if(\$a['a']==1 && \$a['b']==1 && \$a['c']==1){return 'OK';}else{return 'ERR';}}";eval($str);if(a(

python之matplotlib学习绘制动态更新图实例代码

简介 通过定时器Timer触发事件,定时更新绘图,可以形成动态更新图片.下面的实例是学习<matplotlib for python developers>一文的笔记. 实现 实现代码及简单介绍 通过self.user = self.user[1:] + [temp],每次删除列表的第一元素,在其尾部添加新的元素.这样完成user数据的动态更新.其他详细的解释见文中的注释部分. #-*-coding:utf-8-*- import wx from matplotlib.figure impor

vue elementui el-form rules动态验证的实例代码详解

一.介绍 简介:在使用elementUI el-form 中,对于业务不同的时候可能会产生不同表单结构,但是都是存在同一个表单控件el-form中. 图片介绍: 1.在用户选择单选或多选时会有A,B,C,D,E五个选项 2.在用户选择简答题时只会题干,答案以及解析选项(主要是通过v-if来进行判断) 问题引入:当用户选择不同的题库时会产生不同的form表单选项,这个时候在进行表单提交验证的时候就需要根据不同试题类型进行判断,这个时候就需要两个rules来动态进行表单校验. 解决方法:在页面加载的

vue动态注册组件实例代码详解

写本篇文章之前其实也关注过vue中的一个关于加载动态组件is的API,最开始研究它只是用来实现一个tab切换的功能,使用起来也蛮不错的. is 预期:string | Object (组件的选项对象) 用于动态组件且基于 DOM 内模板的限制来工作. 示例: <!-- 当 `currentView` 改变时,组件也跟着改变 --> <component v-bind:is="currentView"></component> 详见vue API中关于

VBScript 动态 Array 的实现代码

记录一个小方法,关于 VBScript 中,动态 Array 的实现,也适用于 VBA, 在很久以前,写 VBA 的时候,就觉得使用 Array 和不方便,因为大小固定, 当时想的是,要是 Array 可以像 Python 里的 list 一样好用该多好啊, 那么下面,就记录一个方法,能让 Array 变得动态,并且好用! 实现方法: 在下面的实例中,先设定一个空的 Array 出来, 然后用,下面的方法实现动态 Array, 并且,把数字 1 到 10,一个加到 Array 中去. '动态 A

Bootstrap jquery.twbsPagination.js动态页码分页实例代码

Bootstrap风格的分页控件自适应的: 参考网址:分页参考文档 1.风格样式: 2.首先引入js文件jQuery.twbsPagination.js <span style="font-size:14px;"><script type="text/javascript" src="plugins/page/jquery.twbsPagination.js"></script></span> 3.

原生js实现class的添加和删除简单代码

实例代码: function hasClass( elements,cName ){ return !!elements.className.match( new RegExp( "(\\s|^)" + cName + "(\\s|$)") ); }; function addClass( elements,cName ){ if( !hasClass( elements,cName ) ){ elements.className += " "

jquery实现的动态回到顶部特效代码

本文实例讲述了jquery实现的动态回到顶部特效代码.分享给大家供大家参考,具体如下: 这款jquery动态回到顶部特效,不是一下子就回到了网页顶部,而是带点缓冲的效果,有动画效果,使用了jQuery插件,这是个非常常用的网页特效,希望大家喜欢哦. 运行效果截图如下: 在线演示地址如下: http://demo.jb51.net/js/2015/jquery-back-top-dh-style-codes/ 具体代码如下: <!DOCTYPE html PUBLIC "-//W3C//DT