【Rust 实战】Rust与C#交互-生成DLL库

简介: 【Rust 实战】Rust与C#交互-生成DLL库

0x00 开篇(Intro)


Rust的强大之处只有你想不到,没有它做不到。既然是打折取代C++语言的旗号,那么肯定只要C++能做到的,他也可以做到。这篇教程来说一下如何使用Rust来创建一个DLL库,并且从C#来调用它。


0x01 所需软件(Software)


  • CLion
  • Visual Studio 2013

注:CLion可以使用VSCode,甚至是记事本来代替,本教程以CLion为例。Visual Studio 2013以上的版本即可。


0x02 编写Rust库(Coding Rust)


创建项目


首先使用CLion创建一个rust lib


0a2653c851af460fa595bd959398a8f1.png


我们直接点开Cargo.toml,按照如下配置添加。


[lib]
name = "TestDLL" #生成dll的文件名
crate-type = ["cdylib"]

这里是配置这个项目生成一个lib库。其中,name是最终生成的DLL库的名称,可以随便起名,我这里按照C#的命名规则来命名为TestDLL。crate-type设置为cdylib。关于为什么这里是cdylib,第四节会解释原因。


简单函数编写


创建完成后,CLion会自动生成单元测试代码,我们可以先直接注释掉。我们先从简单的写起。题目:编写一个函数,输出hello Rust dLL!。代码如下:

#[no_mangle]
pub extern fn hello() {
    println!("hello Rust DLL!");
}

其中,#[no_mangle]为了编译时函数方法名不会被混淆。extern表示该函数是一个外部函数接口。


编译DLL库


控制台直接输入cargo build --release编译。如下图,可以找到DLL文件的位置。


2d65d23f6d4748949b924e4057485923.png


0x03 编写C#项目(Codding C#)


创建项目


简单创建一个C#控制台项目RustDLLTest,如下图所示。


6de278e6d6694ce5bb08e7e842b7e74b.png


导入DLL库


C#导入DLL库的方式有很多种,但是使用Rust生成的DLL库,只能使用DllImoport来导入。具体原因,第四节给出解释。

class Program
    {
        [DllImport("TestDLL.dll", EntryPoint = "hello", CallingConvention = CallingConvention.Cdecl)]
        public static extern void hello();
        static void Main(string[] args)
        {
            hello();
            Console.ReadLine();
        }
    }

DllImport有三个参数,第一个是dll的路径,由于我将dll放到了和生成exe的目录同目录下(如下图),这里就使用了相对路径。第二个参数是ll的入口点,也就是Rust中的方法名。第三个CallingConvention = CallingConvention.Cdecl表示C调用约定。


8ec4f2997fb246878c34ecd6d122b7c6.png


运行程序


直接运行程序,控制台出现hello Rust DLL!。大功告成。可能有部分小伙伴无法运行成功,会碰到BadImageFormatException。第四节将给出解决办法。


12c3b7f3f8814309a195c64f051d4445.png


0x04 答疑解惑(QA)


什么是cdylib


cdylib,是C Dynamic Library的简写,名为C规范动态库。可以生成被其它语言调用的库,也就是跨语言 FFI 使用。因为几乎所有语言都有遵循 C 规范的 FFI 实现,它会按照平台生成.so.dylib.dll等库。当然crate-type还有其它类型,这里暂不介绍了。


为什么只能使用DllImport导入?


DllImport的是标准的dll,可以是DELPHI、C++等各种语言写的标准dll。如果调用C语言等语言编写的普通dll,那么就要用DllImport,典型的像Windows API函数都是C语言编写的dll所以都要DllImport。项目引入的dll,是.NET的dll,它属于非标准dll,只是一个类库。


为什么会报BadImageFormatException?


34e8d716411043c08c7ffba9fbba23de.png


这是由于架构不同的原因。这是一个坑。现在大部分电脑都是64位的架构,而Rust默认生成与电脑架构相同的类库。但是由于C#创建的程序默认是32位架构,导致运行时报错。有以下两种解决办法:


1、Rust生成x86架构的dll库


编译时指定架构,下面的代码是生成x86结构的dll库。

cargo build --release --target i686-pc-windows-msvc

2、指定C# 程序为64位架构

修改项目Properties中的目标平台为x64。


92ba0822ed0b46e1ae72df8a17d3a45b.png


两种办法都可以,选择其一即可。

相关文章
|
4天前
|
C# 数据库
C# 使用 DbDataReader 来访问数据库
C# 使用 DbDataReader 来访问数据库
13 2
|
1月前
|
算法 C#
C#实战 | 求解《九章算术》盈不足之共买物
【7月更文挑战第8天】中国古代数学成就显著,《九章算术》展示了先进的算法,如分数运算和方程解法,领先世界数百年。项目示例通过控制台应用,运用for循环和if条件语句,模拟解决书中盈不足问题,展示了解决数学问题的编程方法。
30 6
C#实战 | 求解《九章算术》盈不足之共买物
|
1月前
|
算法 C#
C#实战 | 求解《丘建算经》百鸡问题
【7月更文挑战第9天】《丘建算经》的百鸡问题是一个经典的不定方程问题,用C#解决时,通过三重嵌套循环穷举公鸡、母鸡和小鸡的组合。代码示例中,外层循环分别对应公鸡和母鸡,而小鸡数量由总钱数和已知鸡种计算得出,避免了额外的内层循环。使用`if`判断确保总数量正确。注意,除法运算可能导致整数截断错误,需使用3.0保证浮点数除法的准确性。这种方法虽然效率较低,但能确保找到所有可行解。
23 1
C#实战 | 求解《丘建算经》百鸡问题
|
5天前
|
数据库 C#
C# 使用SqlDataAdapter和DataSet来访问数据库
C# 使用SqlDataAdapter和DataSet来访问数据库
10 0
|
5天前
|
C#
C# WPF 将第三方DLL嵌入 exe
C# WPF 将第三方DLL嵌入 exe
10 0
|
6天前
|
C# 图形学 数据安全/隐私保护
Unity数据加密☀️ 二、使用Rider将C#代码生成DLL文件
Unity数据加密☀️ 二、使用Rider将C#代码生成DLL文件
|
1月前
|
运维 C# 开发工具
C#实战 | 天行健、上下而求索
【7月更文挑战第7天】使用C语言实现了一个小球(小方块)在屏幕上斜向移动并反弹的程序。当C#入门案例包括创建控制台应用和Windows窗体应用。 1. **控制台应用“天行健,君子以自强不息”** - 使用Visual Studio创建新C#控制台项目,命名为ConsoleAppStrengthenSelf。 - 在Main()方法中使用`Console.WriteLine()`输出励志语句。 - 运行程序,控制台显示结果。 每个项目都涉及Visual Studio的使用,Main()作为程序入口,以及不同类型的用户交互:控制台的文本输出和Windows窗体的图形界面。
31 0
C#实战 | 天行健、上下而求索
|
2月前
|
Rust 图形学
【unity实战】使用unity制作一个类似Rust的3D生存建造建筑系统,具有很好的吸附性(附项目源码)
【unity实战】使用unity制作一个类似Rust的3D生存建造建筑系统,具有很好的吸附性(附项目源码)
44 1
|
1月前
|
机器学习/深度学习 算法 搜索推荐
一个开源且全面的C#算法实战教程
一个开源且全面的C#算法实战教程
|
2月前
|
安全 编译器 API
程序与技术分享:C#调用DLL的几种方法
程序与技术分享:C#调用DLL的几种方法
30 0