文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

P/Invoke之C#调用动态链接库DLL的方法是什么

2023-07-05 19:00

关注

这篇文章主要介绍了P/Invoke之C#调用动态链接库DLL的方法是什么的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇P/Invoke之C#调用动态链接库DLL的方法是什么文章都会有所收获,下面我们一起来看看吧。

P/Invok是什么?

本编所涉及到的工具以及框架:

Visual Studio 2022

.net 6.0

P/Invoke全称为Platform Invoke(平台调用),其实际上就是一种函数调用机制,通过P/Invoke就可以实现调用非托管Dll中的函数。

在开始之前,我们首先需要了解C#中有关托管与非托管的区别

托管(Collocation),即在程序运行时会自动释放内存;

非托管,即在程序运行时不会自动释放内存。

废话不多说,直接实操

第一步:

P/Invoke之C#调用动态链接库DLL的方法是什么

P/Invoke之C#调用动态链接库DLL的方法是什么

在新建的DLL中我们新建一个头文件,用于编写我们的方法定义,然后再次新建一个C++文件,后缀以.c 结尾

P/Invoke之C#调用动态链接库DLL的方法是什么

第二步:

在我们DLL中的头文件(Native.h)中定义相关的Test方法,具体代码如下:

#pragma once// 定义一些宏#ifdef __cplusplus#define EXTERN extern "C"#else#define EXTERN#endif#define CallingConvention _cdecl// 判断用户是否有输入,从而定义区分使用dllimport还是dllexport#ifdef DLL_IMPORT #define HEAD EXTERN __declspec(dllimport)#else#define  HEAD EXTERN __declspec(dllexport)#endifHEAD int CallingConvention Sum(int a, int b);

之后需要去实现头文件中的方法,在Native.c中实现,具体实现如下:

#include "Native.h" // 导入头部文件#include "stdio.h"HEAD int Add(int a, int b){    return a+b;}

P/Invoke之C#调用动态链接库DLL的方法是什么

第三步:

在这里之后,就可以在C#中去尝试调用刚刚所声明的方法,以便验证是否调用DLL成功,其具体实现如下:

using System.Runtime.InteropServices;class Program{    [DllImport(@"C:\My_project\C#_Call_C\CSharp_P_Invoke_Dll\x64\Debug\NativeDll.dll")]    public static extern int Add(int a, int b);    public static void Main(string[] args)    {        int sum = Add(23, 45);        Console.WriteLine(sum);        Console.ReadKey();    }}

运行结果为:68,证明我们成功调用了DLL动态链库

C#中通过P/Invoke调用DLL动态链库的流程

  通过上述一个简单的例子,我们大致了解到了在C#中通过P/Invoke调用DLL动态链库的流程,接下我们将对C#中的代码块做一些改动,便于维护

在改动中我们将用到NativeLibrary类中的一个方法,用于设置回调,解析从程序集进行的本机库导入,并实现通过设置DLL的相对路径进行加载,其方法如下:

public static void SetDllImportResolver (System.Reflection.Assembly assembly, System.Runtime.InteropServices.DllImportResolver resolver);

在使用这个方法前,先查看一下其参数

a、assembly: 主要是获取包含当前正在执行的代码的程序集(不过多讲解)
b、resolber: 此参数是我们要注重实现的,我们可以通过查看他的元代码,发现其实现的是一个委托,因此我们对其进行实现。
原始方法如下:

public delegate IntPtr DllImportResolver(string libraryName, Assembly assembly, DllImportSearchPath? searchPath);

实现resolver方法:

const string NativeLib = "NativeDll.dll";static IntPtr DllImportResolver(string libraryName, Assembly assembly, DllImportSearchPath? searchPath){    string dll = Path.Combine(new DirectoryInfo(Environment.CurrentDirectory).Parent.Parent.Parent.Parent.ToString(), "x64","Release", "NativeDll.dll"); // 此处为Dll的路径    //Console.WriteLine(dll);    return libraryName switch    {        NativeLib => NativeLibrary.Load(dll, assembly, searchPath),        _ => IntPtr.Zero    };}

该方法主要是用于区分在加载DLL时不一定只能是设置绝对路径,也可以使用相对路径对其加载,本区域代码是通过使用委托去实现加载相对路径对其DLL加载,这样做的好处是,便于以后需要更改DLL的路径时,只需要在这个方法中对其相对路径进行修改即可。

更新C#中的代码,其代码如下:

using System.Reflection;using System.Runtime.InteropServices;class Program{    const string NativeLib = "NativeDll.dll";    [DllImport(NativeLib)]    public static extern int Add(int a, int b);    static IntPtr DllImportResolver(string libraryName, Assembly assembly, DllImportSearchPath? searchPath)    {        string dll = Path.Combine(new DirectoryInfo(Environment.CurrentDirectory).Parent.Parent.Parent.Parent.ToString(), "x64","Release", "NativeDll.dll");        Console.WriteLine(dll);        return libraryName switch        {            NativeLib => NativeLibrary.Load(dll, assembly, searchPath),            _ => IntPtr.Zero        };    }    public static void Main(string[] args)    {        NativeLibrary.SetDllImportResolver(Assembly.GetExecutingAssembly(), DllImportResolver);        int sum = Add(23, 45);        Console.WriteLine(sum);        Console.ReadKey();    }}

最后重新编译,检查其是否能顺利编译通过,最终我们的到的结果为:68

至此,我们就完成了一个简单的C#调用动态链接库的案例

  下面将通过一个具体实例,讲述为什么要这样做?(本实例通过从性能方面进行对比)

在DLL中的头文件中,加入如下代码:

HEAD void CBubbleSort(int* array, int length);

在.c文件中加入如下代码:

HEAD void CBubbleSort(int* array, int length){    int temp = 0;    for (int i = 0; i < length; i++)    {        for (int j = i + 1; j < length; j++)        {            if (array[i] > array[j])            {                temp = array[i];                array[i] = array[j];                array[j] = temp;            }        }    }}

C#中的代码修改:

using System.Diagnostics;using System.Reflection;using System.Runtime.InteropServices;class Program{    const string NativeLib = "NativeDll.dll";    [DllImport(NativeLib)]    public unsafe static extern void CBubbleSort(int* arr, int length);    static IntPtr DllImportResolver(string libraryName, Assembly assembly, DllImportSearchPath? searchPath)    {        string dll = Path.Combine(new DirectoryInfo(Environment.CurrentDirectory).Parent.Parent.Parent.Parent.ToString(), "x64", "Release", "NativeDll.dll");        //Console.WriteLine(dll);        return libraryName switch        {            NativeLib => NativeLibrary.Load(dll, assembly, searchPath),            _ => IntPtr.Zero        };    }    public unsafe static void Main(string[] args)    {        int num = 1000;        int[] arr = new int[num];        int[] cSharpResult = new int[num];        //随机生成num数量个(0-10000)的数字        Random random = new Random();        for (int i = 0; i < arr.Length; i++)        {            arr[i] = random.Next(10000);        }        //利用冒泡排序对其数组进行排序        Stopwatch sw = Stopwatch.StartNew();        Array.Copy(arr, cSharpResult, arr.Length);        cSharpResult = BubbleSort(cSharpResult);        Console.WriteLine($"\n C#实现排序所耗时:{sw.ElapsedMilliseconds}ms\n");        // 调用Dll中的冒泡排序算法        NativeLibrary.SetDllImportResolver(Assembly.GetExecutingAssembly(), DllImportResolver);        fixed (int* ptr = &arr[0])        {            sw.Restart();            CBubbleSort(ptr, arr.Length);        }        Console.WriteLine($"\n C实现排序所耗时:{sw.ElapsedMilliseconds}ms");        Console.ReadKey();    }    //冒泡排序算法    public static int[] BubbleSort(int[] array)    {        int temp = 0;        for (int i = 0; i < array.Length; i++)        {            for (int j = i + 1; j < array.Length; j++)            {                if (array[i] > array[j])                {                    temp = array[i];                    array[i] = array[j];                    array[j] = temp;                }            }        }        return array;    }}

执行结果:

C#实现排序所耗时: 130ms
C实现排序所耗时:3ms

关于“P/Invoke之C#调用动态链接库DLL的方法是什么”这篇文章的内容就介绍到这里,感谢各位的阅读!相信大家对“P/Invoke之C#调用动态链接库DLL的方法是什么”知识都有一定的了解,大家如果还想学习更多知识,欢迎关注编程网行业资讯频道。

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯