原文:C#开发奇技淫巧二:根据dll文件加载C++或者Delphi插件

这两天忙着把框架改为支持加载C++和Delphi的插件,来不及更新blog了。
      原来的写的框架只支持c#插件,这个好做,直接用c#的反射功能便可。但是公司不是所有人都搞C#,也不是所有的程序C#都能很好的完成,又或者其他公司提供的API不是C#的,这个时候,就需要这个框架能够支持多种语言了。
      废话不多说,进入正题。
     上网一搜,C#加载非托管的dll,无非就是使用 DllImportAttribute 。然而,这个属性里面要指明dll所在的路径,因为又是写在属性中,因此是在编译的时候就已经把路径写死了,不能动态指定路径加载。
     于是又找了下,终于发现了c#中的一个函数:Marshal.GetDelegateForFunctionPointer。这个函数的功能就是将非托管的函数指针转换为委托。至此,任务完成。Dll的功能无非提提供各种函数,组成所谓的API,有了上述的方法之后,在C#中定义相关的委托(方法的参数列表和参数类型要跟非托管的Dll的参数类型和参数列表对应,具体的对应请google),然后调用上述方法,将非托管的dll转换为相应的委托,这样就能调用非托管的dll了。
      在C#中,我们定义相关的接口,在方法实现中调用相应的委托,这样,一个插件对象就完成了。下面给出相应的类库和使用实例。

 1     public class LoadDll 2     { 3         #region Win32 API : Load dll 4         [DllImport("kernel32.dll")] 5         public static extern IntPtr LoadLibrary(string path); 6  7         [DllImport("kernel32.dll")] 8         public static extern IntPtr GetProcAddress(IntPtr lib, string funcName); 9          [DllImport("kernel32.dll")]         public static extern bool FreeLibrary(IntPtr lib);          [DllImport("kernel32.dll")]         public static extern IntPtr GetStdHandle(int nStdHandle);          [DllImport("user32", EntryPoint = "CallWindowProc")]         public static extern int CallWindowProc(IntPtr lpPreWndFunc, int hwnd, int msg, int wParam, int lParam);         #endregion          private IntPtr _dllLib;          /// <summary>         /// Initializes a new instance of the <see cref="LoadDll"/> class.         /// </summary>         public LoadDll()         {                      }          /// <summary>         /// Initializes a new instance of the <see cref="LoadDll"/> class.         /// </summary>         /// <param name="path">The path.</param>         public LoadDll(string path)         {             _dllLib = LoadLibrary(path);         }          /// <summary>         /// 注销对象时释放资源         /// <see cref="LoadDll"/> is reclaimed by garbage collection.         /// </summary>         ~LoadDll()         {             FreeLibrary(_dllLib);         }          /// <summary>         /// 初始化dll的路径         /// </summary>         /// <param name="path">The path.</param>         public void InitPath(string path)         {             if (_dllLib == IntPtr.Zero)                 _dllLib = LoadLibrary(path);         }          /// <summary>         /// 根据非托管的dll中的方法名称映射成托管的委托类型         /// </summary>         /// <param name="methodName">非托管的dll中的方法名称</param>         /// <param name="methodType">托管的委托类型</param>         /// <returns></returns>         public Delegate InvokeMethod(string methodName, Type methodType)         {             //获取非托管的dll中方法的地址             var methodPtr = GetProcAddress(_dllLib, methodName);             //将非托管的方法转换为委托             return Marshal.GetDelegateForFunctionPointer(methodPtr, methodType);         }     }

调用:

 loadDll = new LoadDll(path);             stop = (StopHandler)loadDll.InvokeMethod("stop", typeof(StopHandler));             start = (StartHandler)loadDll.InvokeMethod("start", typeof(StartHandler));             init = (InitHandler)loadDll.InvokeMethod("init", typeof(InitHandler));             query = (QueryHandler)loadDll.InvokeMethod("query", typeof(QueryHandler));             setDatabaseInfo = (SetDatabaseInfoHandler)loadDll.InvokeMethod("setDatabaseInfo", typeof(SetDatabaseInfoHandler));             setMonitorInfo = (SetMonitorInfoHandler)loadDll.InvokeMethod("setMonitorInfo", typeof(SetMonitorInfoHandler));

c++中的导出方法:

 /// 插件导出方法 extern "C" __declspec(dllexport) void setDatabaseInfo(LPCTSTR dbServer, LPCTSTR dbUser, LPCTSTR dbPass, LPCTSTR dbName); extern "C" __declspec(dllexport) void setMonitorInfo(LPCTSTR _agentBm, LPCTSTR _com); extern "C" __declspec(dllexport) void init(); extern "C" __declspec(dllexport) void start(); extern "C" __declspec(dllexport) void stop(); extern "C" __declspec(dllexport) LPCTSTR query(LPCTSTR devname, LPCTSTR id);

C#中的委托

 ///<summary>         /// 处理Stop事件         ///</summary>         publicdelegatevoid StopHandler();         ///<summary>         /// 处理Start事件         ///</summary>         publicdelegatevoid StartHandler();         ///<summary>         /// 处理Init事件         ///</summary>         publicdelegatevoid InitHandler();         ///<summary>         /// 处理Query事件         ///</summary>         publicdelegatestring QueryHandler(string devName, string id);         ///<summary>         /// 处理SetDataBaseInfo事件         ///</summary>         publicdelegatevoid SetDatabaseInfoHandler(string server,string user,string password,string dbName);         ///<summary>         /// 处理SetMonitorInfo事件         ///</summary>         publicdelegatevoid SetMonitorInfoHandler(string agentBm, string com);

接下来怎么搞,你们都懂的

C#开发奇技淫巧二:根据dll文件加载C++或者Delphi插件的更多相关文章

  1. 一步一步开发Game服务器(三)加载脚本和服务器热更新(二)完整版

    上一篇文章我介绍了如果动态加载dll文件来更新程序 一步一步开发Game服务器(三)加载脚本和服务器热更新 可是在使用过程中,也许有很多会发现,动态加载dll其实不方便,应为需要预先编译代码为dll文 ...

  2. 给.DLL文件加一个数字签名的方法

     给.dll文件加一个数字签名的方法 效果如图所示:   做法: 下载数字签名工具包:http://files.cnblogs.com/babyt/SignTool.rar  /Files/JavaC ...

  3. 伟景行 citymaker 从入门到精通(1)——js开发,最基本demo,加载cep工程文件

    开发环境:citymaker 7(以下简称cm),jquery,easyui 1.4(界面),visual studio 2012(没有vs,不部署到IIS也行,html文件在本地目录双击打开可用) ...

  4. dll的加载方式主要分为两大类,显式和隐式链接

    之前简单写过如何创建lib和dll文件及简单的使用(http://blog.csdn.net/betabin/article/details/7239200).现在先再深入点写写dll的加载方式. d ...

  5. VSTO学习笔记(三) 开发Office 2010 64位COM加载项

    原文:VSTO学习笔记(三) 开发Office 2010 64位COM加载项 一.加载项简介 Office提供了多种用于扩展Office应用程序功能的模式,常见的有: 1.Office 自动化程序(A ...

  6. Android 的 so 文件加载机制

    本篇文章已授权微信公众号 guolin_blog (郭霖)独家发布 最近碰到一些 so 文件问题,顺便将相关知识点梳理一下. 提问 本文的结论是跟着 System.loadlibrary() 一层层源 ...

  7. 固定dll的加载基址的方法

    调试dll的时候会有一件事情比较烦人,就是dll加载的地址不会很固定(默认设置下编译的dll基址总是0x10000000,多个同基址的dll加载时,后面的肯定会被重定位),这给前后多次调试时对比分析结 ...

  8. php文件加载、错误处理、方法函数和数组

    数组运算符注意:php中,数组的元素的顺序,不是由下标(键名)决定的,而是完全由加入的顺序来决定.联合(+):将右边的数组项合并到左边数组的后面,得到一个新数组.如有重复键,则结果以左边的为准$v1 ...

  9. 插件化框架解读之so 文件加载机制(四)

    阿里P7移动互联网架构师进阶视频(每日更新中)免费学习请点击:https://space.bilibili.com/474380680 提问 本文的结论是跟着 System.loadlibrary() ...

随机推荐

  1. 你不知道的JavaScript演示代码Github地址

    你不知道的JavaScript博文相关代码托管至Github,每次写完博客会把代码提交上去. 代码地址:https://github.com/rongbo-j/you-dont-know-js 点击D ...

  2. FCC高级编程篇之Make a Person

    Make a Person Fill in the object constructor with the following methods below: getfirstname() getLas ...

  3. CSS3中的transition

    W3C标准中对CSS3的transition是这样描述的: CSS的transition允许CSS的属性值在一定的时间区间内平滑地过渡.这种效果可以在鼠标单击,获得焦点,被点击或对元素任何改变中触发, ...

  4. JMS消息

    1.消息可分为3部分:消息头.属性和有效负载 消息头:用于标识消息.声明消息属性及提供路由信息的特殊字段组成. 消息的属性区包含了和该消息有关的附加元数据,这个元数据由应用程序开发者进行设置,或者由J ...

  5. django orm 基本

    1 modle基本数据类型 class Test(models.Model): """测试学习用""" Auto = models.Auto ...

  6. 常见VPS buy地址

    ***,也是最适合新手使用的: https://bwh1.net/ (支持支付宝) vultr,以下是我的分享链接: https://www.vultr.com/(支持支付宝) SugarHosts: ...

  7. 洛谷 P2747 [USACO5.4]周游加拿大Canada Tour

    P2747 [USACO5.4]周游加拿大Canada Tour 题目描述 你赢得了一场航空公司举办的比赛,奖品是一张加拿大环游机票.旅行在这家航空公司开放的最西边的城市开始,然后一直自西向东旅行,直 ...

  8. [Angular] Configure an Angular App at Compile Time with the Angular CLI

    Compile time configuration options allow you to provide different kind of settings based on the envi ...

  9. linux下创建带password的用户

    一直在做实验室linuxserver的账号管理系统,现阶段是用户申请后我这边收到邮件,然后手动创建,这个略显麻烦,打算全然做成自己主动化的.用户申请后,我直接在管理界面点击批准就可以创建用户,同一时候 ...

  10. Caused by: java.lang.ClassNotFoundException: com.njupt.libgdxbase.MainActivity

    在使用libgdx来开发游戏时.假设遇到这样的问题.非常可能是由于你没有在libgdx的项目中导入Android的现骨干jar包导致的. 解决方法例如以下: 右击项目---"build pa ...