Rust语言学习

Rust与C#交互——生成DLL库

2022-01-08  本文已影响0人  L我是小学生

0x00 开篇(Intro)

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

0x01 所需软件(Software)

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

0x02 编写Rust库(Coding Rust)

创建项目

首先使用CLion创建一个rust lib

image
我们直接点开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文件的位置。

image-20220108132949115

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

创建项目

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

image
导入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调用约定。

image
运行程序

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

image

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?
image

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

1、Rust生成x86架构的dll库

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

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

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

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

image

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

0x05 源码(Source Code)

关注公众号《Rust学习日记》,回复【dll】获取完整源码。

上一篇下一篇

猜你喜欢

热点阅读