使用.Net Core RT 标准动态库
这个文档可以引导你如何通过CoreRT生成一个原生标准的系统动态库让其他编程语言调用. CoreRT 可以构建静态库, 这些库可以在编译时链接或者也可以构建运行时所需的共享库,
创建一个支持CoreRT的 .NET Core 类库项目
使用 dotnet new console -o NativeLibrary
创建一个.NET Core类库项目并参考Hello world 示例为项目加入 CoreRT 的支持.
生成静态库
> dotnet publish /p:NativeLib=Static -r <RID> -c <Configuration>
<Configuration>
是你项目的配置 (比如Debug 或者 Release) 和 <RID>
是运行时标识(可以是 win-x64, linux-x64, osx-x64其中之一).比如,你想发布一个 release 配置并且是Windows 64位动态库的时候命令行你可以这么写:
> dotnet publish /p:NativeLib=Static -r win-x64 -c release
上面的命令将删掉一个静态库 (Windows .lib
, OSX/Linux .a
) 用 ./bin/[configuration]/netstandard2.0/[RID]/publish/
文件夹并将有一个同名文件夹在你的源码文件夹呈现。
生成共享库
> dotnet publish /p:NativeLib=Shared -r <RID> -c <Configuration>
上面的命令将删掉共享库 (Windows .dll
, OSX .dylib
, Linux .so
) 在 ./bin/[configuration]/netstandard2.0/[RID]/publish/
文件夹并将有一个同名文件夹在你的源码中呈现.生成共享库在Linux上目前截止2018年12月6日不能正常工作, 详情查看 #4988.
导出方法
针对一个 C# 方法再本地原生动态库被外部程序调用, 必须要使用 [NativeCallable]
属性进行导出. 首先定义NativeCallable
类在哪i的项目中,请查看这里.本地定义 NativeCallable
是一个临时解决方法,正式发布的时候会添加到.Net Core 内.
下一步,在要导出的方法上使用 EntryPoint
和 CallingConvention
属性:
[NativeCallable(EntryPoint = "add", CallingConvention = CallingConvention.StdCall)]
public static int Add(int a, int b)
{
return a + b;
}
在原生动态库生成后, C# 的Add
方法被导出为add
函数功其他语言调用. 在决定要导出的托管方法时, 需要考虑以下一些限制:
- 导出方法必须是静态.
- 导出方法只能是原生可接收或值类型(比如结构 ),他们必须封装所有引用类型参数 .
- 无法从常规托管 c# 代码调用导出的方法, 会有发生异常.
- 导出的方法不能使用常规的 C# 异常捕获,必须使用错误代码替换.
引用
真实案例如何使用 CoreRT编写用于Rust的原始动态库: https://medium.com/@chyyran/calling-c-natively-from-rust-1f92c506289d