C#动态加载第三方非托管DLL,LoadLibraryEx,LoadLibrary

本文主要是介绍C#动态加载第三方非托管DLL,LoadLibraryEx,LoadLibrary,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

C#动态加载第三方DLL

当我们需要加载第三方非托管DLL时,通常会直接使用DllImport的方式,代码如下:

  1. [DllImport("GetFile.dll", CallingConvention = CallingConvention.StdCall, CharSet = CharSet.Unicode)]

  2. static extern string GetFileData(string fileName);

上图的调用方式,默认GetFile.dll文件位于与调用程序(.exe文件)相同的目录中(这里不考虑System32目录、环境变量目录,因为通常情况下,不会将第三方DLL放到这些目录中)。

如果不想将DLL放到exe所在目录,那也可以手动指定DLL文件路径,代码如下:

  1. [DllImport("C:\\Customer\\GetFile.dll", CallingConvention = CallingConvention.StdCall, CharSet = CharSet.Unicode)]

  2. static extern string GetFileData(string fileName);

更进一步,如果此时我们想动态指定DLL文件路径,则以上方式将无法应对,原因是DllImport中的DLL文件路径必须是常量。

为了动态调用DLL,我们需要通过其它方式,具体代码如下。这里我们定义了一个DllInvoke类,其中用到了LoadLibrary()函数,通过该函数导入DLL(如上文中的GetFile.dll)文件,然后再通过GetProcAddress()函数获取DLL中欲使用的API(上文中为GetFileData())的指针,最终通过Marshal.GetDelegateForFunctionPointer()函数返回API对应的委托。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Runtime.InteropServices;class DllInvoke
{[DllImport("kernel32.dll", SetLastError = true)]private static extern IntPtr LoadLibrary(string lpFileName);[DllImport("kernel32.dll")]private extern static IntPtr GetProcAddress(IntPtr lib, String funcName);[DllImport("kernel32.dll")]private extern static bool FreeLibrary(IntPtr lib);private IntPtr hLib;public DllInvoke(String DllName){hLib = LoadLibrary(DllName);if (hLib == IntPtr.Zero){var err = Marshal.GetLastWin32Error(); //只有SetLastError = true时,才能获取到Error Code}}~DllInvoke(){FreeLibrary(hLib);}//将要执行的函数转换为委托public Delegate Invoke(String ApiName, Type t){IntPtr api = GetProcAddress(hLib, ApiName);return (Delegate)Marshal.GetDelegateForFunctionPointer(api, t);}
}

另外,有个问题需要注意。当被调用的DLL,它自身也在调用其它DLL时,此时使用上文中的LoadLibrary()函数时,会有值为126的Error Code(通过Marshal.GetLastWin32Error()函数获取Error Code,126表示找不到指定的模块)。之所以会有此报错,是因为对于被调用DLL自身调用的“其它DLL”,会按照和DllImport同样的顺序(即exe所在目录、System32目录、环境变量目录)去寻找它们,而这些“其它DLL”你可能放在了和被调用DLL同样的目录中,而没有放在exe所在目录、System32目录、环境变量目录中,所以显然是找不到的。

如果我就是想把所有的DLL(不论是被调用的DLL,还是被调用DLL自身调用的其它DLL)都放在一个我自己指定的目录中呢?此时我们可以使用LoadLibraryEx()函数,通过该函数导入的DLL,如果它本身也调用了其它DLL的话,会强制先在被调用DLL所在的目录中查找它调用的“其它DLL”。使用LoadLibraryEx()函数的完整代码如下,新的DllInvoke类中,定义了一个LoadLibraryFlags枚举变量,其中的LOAD_WITH_ALTERED_SEARCH_PATH值会被传入LoadLibraryEx()函数中,从而强制在DLL所在目录中查找。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Runtime.InteropServices;class DllInvoke
{/// <summary>/// LoadLibraryFlags/// </summary>public enum LoadLibraryFlags : uint{DONT_RESOLVE_DLL_REFERENCES = 0x00000001,LOAD_IGNORE_CODE_AUTHZ_LEVEL = 0x00000010,LOAD_LIBRARY_AS_DATAFILE = 0x00000002,LOAD_LIBRARY_AS_DATAFILE_EXCLUSIVE = 0x00000040,LOAD_LIBRARY_AS_IMAGE_RESOURCE = 0x00000020,LOAD_LIBRARY_SEARCH_APPLICATION_DIR = 0x00000200,LOAD_LIBRARY_SEARCH_DEFAULT_DIRS = 0x00001000,LOAD_LIBRARY_SEARCH_DLL_LOAD_DIR = 0x00000100,LOAD_LIBRARY_SEARCH_SYSTEM32 = 0x00000800,LOAD_LIBRARY_SEARCH_USER_DIRS = 0x00000400,LOAD_WITH_ALTERED_SEARCH_PATH = 0x00000008}[DllImport("kernel32.dll", SetLastError = true)]private static extern IntPtr LoadLibraryEx(string lpFileName, IntPtr hReservedNull, LoadLibraryFlags dwFlags);[DllImport("kernel32.dll")]private extern static IntPtr GetProcAddress(IntPtr lib, String funcName);[DllImport("kernel32.dll")]private extern static bool FreeLibrary(IntPtr lib);private IntPtr hLib;public DllInvoke(String DllName){hLib = LoadLibraryEx(DllName, IntPtr.Zero, LoadLibraryFlags.LOAD_WITH_ALTERED_SEARCH_PATH);if (hLib == IntPtr.Zero){var err = Marshal.GetLastWin32Error(); //只有SetLastError = true时,才能获取到Error Code}}~DllInvoke(){FreeLibrary(hLib);}//将要执行的函数转换为委托public Delegate Invoke(String ApiName, Type t){IntPtr api = GetProcAddress(hLib, ApiName);return (Delegate)Marshal.GetDelegateForFunctionPointer(api, t);}
}

如何使用以上定义的DllInvoke类呢?具体代码如下,首先定义一个委托CustomerAPI(该委托的形参列表需要和待调用的DLL中的API一致),然后即可用类似下图Example()函数中的代码,进行API的调用。

public delegate int CustomerAPI(string fileName);
private void Example()
{string dllName = "C:\\Customer\\GetFile.dll";DllInvoke customerDll = new DllInvoke(dllName);CustomerAPI GetFileData = (CustomerAPI)customerDll.Invoke("GetFileData", typeof(CustomerAPI));string fileName = "C:\\Customer\\ExampleFile.txt";int data = GetFileData(fileName);MessageBox.Show("The data got from file is: " + data.ToString());
}

这篇关于C#动态加载第三方非托管DLL,LoadLibraryEx,LoadLibrary的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/515851

相关文章

C#如何去掉文件夹或文件名非法字符

《C#如何去掉文件夹或文件名非法字符》:本文主要介绍C#如何去掉文件夹或文件名非法字符的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C#去掉文件夹或文件名非法字符net类库提供了非法字符的数组这里还有个小窍门总结C#去掉文件夹或文件名非法字符实现有输入字

C#之List集合去重复对象的实现方法

《C#之List集合去重复对象的实现方法》:本文主要介绍C#之List集合去重复对象的实现方法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C# List集合去重复对象方法1、测试数据2、测试数据3、知识点补充总结C# List集合去重复对象方法1、测试数据

C#实现将Office文档(Word/Excel/PDF/PPT)转为Markdown格式

《C#实现将Office文档(Word/Excel/PDF/PPT)转为Markdown格式》Markdown凭借简洁的语法、优良的可读性,以及对版本控制系统的高度兼容性,逐渐成为最受欢迎的文档格式... 目录为什么要将文档转换为 Markdown 格式使用工具将 Word 文档转换为 Markdown(.

springboot加载不到nacos配置中心的配置问题处理

《springboot加载不到nacos配置中心的配置问题处理》:本文主要介绍springboot加载不到nacos配置中心的配置问题处理,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑... 目录springboot加载不到nacos配置中心的配置两种可能Spring Boot 版本Nacos

Java调用C#动态库的三种方法详解

《Java调用C#动态库的三种方法详解》在这个多语言编程的时代,Java和C#就像两位才华横溢的舞者,各自在不同的舞台上展现着独特的魅力,然而,当它们携手合作时,又会碰撞出怎样绚丽的火花呢?今天,我们... 目录方法1:C++/CLI搭建桥梁——Java ↔ C# 的“翻译官”步骤1:创建C#类库(.NET

MyBatis编写嵌套子查询的动态SQL实践详解

《MyBatis编写嵌套子查询的动态SQL实践详解》在Java生态中,MyBatis作为一款优秀的ORM框架,广泛应用于数据库操作,本文将深入探讨如何在MyBatis中编写嵌套子查询的动态SQL,并结... 目录一、Myhttp://www.chinasem.cnBATis动态SQL的核心优势1. 灵活性与可

C#代码实现解析WTGPS和BD数据

《C#代码实现解析WTGPS和BD数据》在现代的导航与定位应用中,准确解析GPS和北斗(BD)等卫星定位数据至关重要,本文将使用C#语言实现解析WTGPS和BD数据,需要的可以了解下... 目录一、代码结构概览1. 核心解析方法2. 位置信息解析3. 经纬度转换方法4. 日期和时间戳解析5. 辅助方法二、L

Mybatis嵌套子查询动态SQL编写实践

《Mybatis嵌套子查询动态SQL编写实践》:本文主要介绍Mybatis嵌套子查询动态SQL编写方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录前言一、实体类1、主类2、子类二、Mapper三、XML四、详解总结前言MyBATis的xml文件编写动态SQL

使用C#删除Excel表格中的重复行数据的代码详解

《使用C#删除Excel表格中的重复行数据的代码详解》重复行是指在Excel表格中完全相同的多行数据,删除这些重复行至关重要,因为它们不仅会干扰数据分析,还可能导致错误的决策和结论,所以本文给大家介绍... 目录简介使用工具C# 删除Excel工作表中的重复行语法工作原理实现代码C# 删除指定Excel单元

使用Python获取JS加载的数据的多种实现方法

《使用Python获取JS加载的数据的多种实现方法》在当今的互联网时代,网页数据的动态加载已经成为一种常见的技术手段,许多现代网站通过JavaScript(JS)动态加载内容,这使得传统的静态网页爬取... 目录引言一、动态 网页与js加载数据的原理二、python爬取JS加载数据的方法(一)分析网络请求1