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

1、Visual Studio 2022

2、.net 6.0

P/Invok是什么?

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

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

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

废话不多说,直接实操

第一步:

  1. 打开VS2022,新建一个C#控制台应用

  1. 右击解决方案,添加一个新建项,新建一个"动态链接库(DLL)",新建完之后需要右击当前项目--> 属性 --> C/C++ --> 预编译头 --> 选择"不使用编译头"

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

第二步:

  1. 在我们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)
    #endif HEAD int CallingConvention Sum(int a, int b);
  2. 之后需要去实现头文件中的方法,在Native.c中实现,具体实现如下:

    #include "Native.h" // 导入头部文件
    #include "stdio.h" HEAD int Add(int a, int b)
    {
    return a+b;
    }
  3. 在这些步骤做完后,可以尝试生成解决方案,检查是否报错,没有报错之后,将进入项目文件中,检查是否生成DLL (../x64/Debug)

第三步:

  1. 在这里之后,就可以在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#中的代码块做一些改动,便于维护

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

    public static void SetDllImportResolver (System.Reflection.Assembly assembly, System.Runtime.InteropServices.DllImportResolver resolver);
  2. 在使用这个方法前,先查看一下其参数

    a、assembly: 主要是获取包含当前正在执行的代码的程序集(不过多讲解)

    b、resolber: 此参数是我们要注重实现的,我们可以通过查看他的元代码,发现其实现的是一个委托,因此我们对其进行实现。

    原始方法如下:

    public delegate IntPtr DllImportResolver(string libraryName, Assembly assembly, DllImportSearchPath? searchPath);
  3. 实现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的路径时,只需要在这个方法中对其相对路径进行修改即可。

  4. 更新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();
    }
    }
  5. 最后重新编译,检查其是否能顺利编译通过,最终我们的到的结果为:68

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

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

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

    HEAD void CBubbleSort(int* array, int length);
  2. 在.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;
    }
    }
    }
    }
  3. 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;
    }
    }
  4. 执行结果:

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

    在实现本案例中,可能在编译后,大家所看到的结果不是很出乎意料,但这只是一种案例,希望通过此案例的分析,能给大家带来一些意想不到的收获叭。

最后

简单做一下总结叭,通过上述所描述的从第一步如何创建一个DLL到如何通过C#去调用的一个简单实例,也应该能给正在查阅相关资料的你有所收获,也希望能给在这方面有所研究的你有一些相关的启发,同时也希望能给目前对这方面毫无了解的你有一个更进一步的学习。

1、Visual Studio 2022

2、.net 6.0

P/Invok是什么?

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

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

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

废话不多说,直接实操

第一步:

  1. 打开VS2022,新建一个C#控制台应用

  2. 右击解决方案,添加一个新建项,新建一个"动态链接库(DLL)",新建完之后需要右击当前项目--> 属性 --> C/C++ --> 预编译头 --> 选择"不使用编译头"

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

第二步:

  1. 在我们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)
    #endif HEAD int CallingConvention Sum(int a, int b);
  2. 之后需要去实现头文件中的方法,在Native.c中实现,具体实现如下:

    #include "Native.h" // 导入头部文件
    #include "stdio.h" HEAD int Add(int a, int b)
    {
    return a+b;
    }
  3. 在这些步骤做完后,可以尝试生成解决方案,检查是否报错,没有报错之后,将进入项目文件中,检查是否生成DLL (../x64/Debug)

第三步:

  1. 在这里之后,就可以在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#中的代码块做一些改动,便于维护

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

    public static void SetDllImportResolver (System.Reflection.Assembly assembly, System.Runtime.InteropServices.DllImportResolver resolver);
  2. 在使用这个方法前,先查看一下其参数

    a、assembly: 主要是获取包含当前正在执行的代码的程序集(不过多讲解)

    b、resolber: 此参数是我们要注重实现的,我们可以通过查看他的元代码,发现其实现的是一个委托,因此我们对其进行实现。

    原始方法如下:

    public delegate IntPtr DllImportResolver(string libraryName, Assembly assembly, DllImportSearchPath? searchPath);
  3. 实现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的路径时,只需要在这个方法中对其相对路径进行修改即可。

  4. 更新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();
    }
    }
  5. 最后重新编译,检查其是否能顺利编译通过,最终我们的到的结果为:68

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

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

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

    HEAD void CBubbleSort(int* array, int length);
  2. 在.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;
    }
    }
    }
    }
  3. 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;
    }
    }
  4. 执行结果:

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

    在实现本案例中,可能在编译后,大家所看到的结果不是很出乎意料,但这只是一种案例,希望通过此案例的分析,能给大家带来一些意想不到的收获叭。

最后

简单做一下总结叭,通过上述所描述的从第一步如何创建一个DLL到如何通过C#去调用的一个简单实例,也应该能给正在查阅相关资料的你有所收获,也希望能给在这方面有所研究的你有一些相关的启发,同时也希望能给目前对这方面毫无了解的你有一个更进一步的学习。

作者:百宝门-刘忠帅

原文地址:https://blog.baibaomen.com/p-invoke之c调用动态链接库dll/

P/Invoke之C#调用动态链接库DLL的更多相关文章

  1. BCB如何编写,调用动态链接库DLL

    一 编写动态链接库DLL DLL简称动态链接库,是Windows中程序的重要组成部分.想象一下,一个程序需要多人共同完成开发,怎么个共同法?这时我们就要考虑把程序分为好几个模块,团队每一个成员开发一个 ...

  2. ASP.net/C#中如何调用动态链接库DLL

    动态链接库(也称为DLL,即为“Dynamic Link Library”的缩写)是Microsoft Windows最重要的组成要素之一,打开Windows系统文件夹,你会发现文件夹中有很多DLL文 ...

  3. LR调用动态链接库DLL

    什么是动态库? 动态库一般又叫动态链接库(DLL),是Dynamic Link Library 的缩写形式,DLL是一个包含可由多个程序同时使用的代码和数据的库. 动态链接提供了一种方法 ,使进程可以 ...

  4. 可执行Jar包调用动态链接库(DLL/SO)

    踩过了很多的坑,查了很多资料,在此记录一下,以SpringBoot项目为基础. Maven加入JNA依赖 <!-- JNA start --> <dependency> < ...

  5. JAVA调用动态链接库DLL之JNative学习

    package com.ehfscliax; import java.io.UnsupportedEncodingException;import java.net.URLEncoder;import ...

  6. JAVA调用动态链接库(dll)

        菜鸡爬坑 基础知识  因为某个东西的keygen我只会在win下生成!! 所以只能出此下策!!之前一直是android下用jni调用so文件,现在试下java在win平台下调用dll 首先还是 ...

  7. 制作和unity调用动态链接库dll文件

    首先用vc建立一个dll工程 然后在里面建立一个testunity.h文件.内容如下 1 extern "C" int _declspec(dllexport)testunity( ...

  8. VS2010编写动态链接库DLL及单元测试用例,调用DLL测试正确性

    转自:http://blog.csdn.net/testcs_dn/article/details/27237509 本文将创建一个简单的动态链接库,并编写一个控制台应用程序使用该动态链接库,该动态链 ...

  9. VS2010编写动态链接库DLL和单元测试,转让DLL测试的正确性

    本文将创建一个简单的动态库-link,谱写控制台应用程序使用该动态链接库,该动态链接库为"JAVA调用动态链接库DLL之JNative学习"中使用的DLL,仅仅是项目及文件名不同. ...

  10. 编译可供C#调用的C/C++动态链接库dll文件

    编译可供C#调用的C/C++动态链接库dll文件,C语言控制台应用程序,探索生成dll过程 由于项目需求,需要公司另一个团队提供相关算法支持,是用C语言编译好的dll库提供给我们进行调用. 但是拿到d ...

随机推荐

  1. python字符串拼接方式

    一 .join(iterator),后面必须是可迭代对象,例如:字符串,列表,元组 testList = ["1","2"] print(",&quo ...

  2. 【APT】APT-C-41下载器组件样本分析

    前言 APT-C-41(又被称为蓝色魔眼.Promethium.StrongPity),该APT组织最早的攻击活动可以追溯到2012年.该组织主要针对意大利.土耳其.比利时.叙利亚.欧洲等地区和国家进 ...

  3. python读取与处理netcdf数据

    netcdf是气候数据中的主流格式,当涉及到大范围的全球数万个格网点数据时,使用python脚本可以较快地读取与处理. import netCDF4 from netCDF4 import Datas ...

  4. CSS设置边距

    1.内边距 所有的 ​HTML ​元素基本都是以矩形为基础. 每个 HTML 元素周围的矩形空间由三个重要的属性来控制: ​padding(内边距)​ ​margin(外边距)​ ​border(边框 ...

  5. Shell脚本结构化-控制流

    脚本结构化命令 上一章给出的那些 shell 脚本里,shell 按照命令在脚本中出现的顺序依次进行处理.对顺序操作来说,这已经足够了,因为在这种操作环境下,你想要的就是所有的命令按照正确的顺序执行. ...

  6. CSS 语法-熟悉样式规则

    CSS 规则: CSS 规则由两个主要的部分构成:选择器,以及一条或多条声明. 样式规则: 以内嵌式样式表为例: (1)所有的css代码都必须书写在<head>标签内部的一对<sty ...

  7. [Leetcode 111]二叉树的最短深度 BFS/DFS

    题目 给定二叉树,求最短路径包含的节点个数 https://leetcode.com/problems/minimum-depth-of-binary-tree/ Given a binary tre ...

  8. IT工具知识-12:RTL8832AU网卡在WIN10更新KB5015807后出现无法正常连接的一种解决方法

    系统配置 硬件配置 使用网卡为Fenvi的FU-AX1800 USB外置网卡(官网驱动同AX1800P) 问题描述 在win10自动更新了KB5015807出现了wifi开机无法自动连接,wifi图标 ...

  9. jquery 时间戳转化为日期时间格式,年月日 时分秒

    <script type="text/javascript"> var strDate = ''; $(function(){ // 获取时间戳 时间戳为10位需*10 ...

  10. 整数划分问题(Java递归)

    整数划分问题(Java递归) 文章目录 整数划分问题(Java递归) 0. 问题描述 1.递归式 2.代码 3.参考 0. 问题描述 整数划分问题 将正整数n表示成一系列正整数之和:n=n1+n2+- ...