百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程网 > 正文

.NET 7 AOT 的使用以及 .NET 与 Go 互相调用

yuyutoo 2025-01-12 19:59 1 浏览 0 评论

目录

  • 背景

  • C# 部分

    • 环境要求

    • 创建一个控制台项目

    • 体验 AOT 编译

    • C# 调用库函数

    • 减少体积

    • C# 导出函数

    • C# 调用 C# 生成的 AOT

  • Golang 部分

    • 安装 GCC

    • Golang 导出函数

  • .NET C# 和 Golang 互调

    • C# 调用 Golang

    • Golang 调用 C#

    • 其他


背景

其实,规划这篇文章有一段时间了,但是比较懒,所以一直拖着没写。

最近时总更新太快了,太卷了,所以借着 .NET 7 正式版发布,熬夜写完这篇文章,希望能够追上时总的一点距离。

本文主要介绍如何在 .NET 和 Go 语言中如何生成系统(Windows)动态链接库,又如何从代码中引用这些库中的函数。

在 .NET 部分,介绍如何使用 AOT、减少二进制文件大小、使用最新的 [LibraryImport] 导入库函数;

在 Go 语言部分,介绍如何使用 GCC 编译 Go 代码、如何通过 syscall 导入库函数。

在文章中会演示 .NET 和 Go 相互调用各自生成的动态链接库,以及对比两者之间的差异。

本文文章内容以及源代码,可以 https://github.com/whuanle/csharp_aot_golang 中找到,如果本文可以给你带来帮助,可以到 Github 点个星星嘛。

C# 部分

环境要求

SDK:.NET 7 SDKDesktop development with C++ workload

IDE:Visual Studio 2022

Desktop development with C++ workload 是一个工具集,里面包含 C++ 开发工具,需要在 Visual Studio Installer 中安装,如下图红框中所示。

创建一个控制台项目

首先创建一个 .NET 7 控制台项目,名称为 CsharpAot

打开项目之后,基本代码如图所示:

我们使用下面的代码做测试:

public class Program
{
static void Main()
{
Console.WriteLine("C# Aot!");
Console.ReadKey();
}
}

体验 AOT 编译

这一步,可以参考官方网站的更多说明:

https://learn.microsoft.com/zh-cn/dotnet/core/deploying/native-aot/

为了能够让项目发布时使用 AOT 模式,需要在项目文件中加上 <PublishAot>true</PublishAot> 选项。

然后使用 Visual Studio 发布项目。

发布项目的配置文件设置,需要按照下图进行配置。

AOT 跟 生成单个文件 两个选项不能同时使用,因为 AOT 本身就是单个文件。

配置完成后,点击 发布,然后打开 Release 目录,会看到如图所示的文件。

.exe 是独立的可执行文件,不需要再依赖 .NET Runtime 环境,这个程序可以放到其他没有安装 .NET 环境的机器中运行。

然后删除以下三个文件:

 CsharpAot.exp
CsharpAot.lib
CsharpAot.pdb

光用 .exe 即可运行,其他是调试符号等文件,不是必需的。

剩下 CsharpAot.exe 文件后,启动这个程序:

C# 调用库函数

这一部分的代码示例,是从笔者的一个开源项目中抽取出来的,这个项目封装了一些获取系统资源的接口,以及快速接入 Prometheus 监控。

不过很久没有更新了,最近没啥动力更新,读者可以点击这里了解一下这个项目:

https://github.com/whuanle/CZGL.SystemInfo/tree/net6.0/src/CZGL.SystemInfo/Memory

因为后续代码需要,所以现在请开启 “允许不安全代码”。

本小节的示例是通过使用 kernel32.dll 去调用 Windows 的内核 API(Win32 API),调用 GlobalMemoryStatusEx 函数 检索有关系统当前使用物理内存和虚拟内存的信息

使用到的 Win32 函数可参考:https://learn.microsoft.com/zh-cn/windows/win32/api/sysinfoapi/nf-sysinfoapi-globalmemorystatusex

关于 .NET 调用动态链接库的方式,在 .NET 7 之前,通过这样调用:

 [DllImport("Kernel32.dll", CharSet = CharSet.Auto, SetLastError = true)]
[return: MarshalAs(UnmanagedType.Bool)]
internal static extern Boolean GlobalMemoryStatusEx(ref MemoryStatusExE lpBuffer);

在 .NET 7 中,出现了新的操作方式 [LibraryImport]

文档是这样介绍的:

Indicates that a source generator should create a function for marshalling arguments instead of relying on the runtime to generate an equivalent marshalling function at run time.

指示源生成器应创建用于编组参数的函数,而不是依赖运行库在运行时生成等效的编组函数。

简单来说,就是我们要使用 AOT 写代码,然后代码中引用到别的动态链接库时,需要使用 [LibraryImport] 引入这些函数。

笔者没有在 AOT 下测试过 [DllImport],读者感兴趣可以试试。

新建两个结构体 MEMORYSTATUS.csMemoryStatusExE.cs

MEMORYSTATUS.cs

public struct MEMORYSTATUS
{
internal UInt32 dwLength;
internal UInt32 dwMemoryLoad;
internal UInt32 dwTotalPhys;
internal UInt32 dwAvailPhys;
internal UInt32 dwTotalPageFile;
internal UInt32 dwAvailPageFile;
internal UInt32 dwTotalVirtual;
internal UInt32 dwAvailVirtual;
}

MemoryStatusExE.cs

public struct MemoryStatusExE
{
/// <summary>
/// 结构的大小,以字节为单位,必须在调用 GlobalMemoryStatusEx 之前设置此成员,可以用 Init 方法提前处理
/// </summary>
/// <remarks>应当使用本对象提供的 Init ,而不是使用构造函数!</remarks>
internal UInt32 dwLength;

/// <summary>
/// 一个介于 0 和 100 之间的数字,用于指定正在使用的物理内存的大致百分比(0 表示没有内存使用,100 表示内存已满)。
/// </summary>
internal UInt32 dwMemoryLoad;

/// <summary>
/// 实际物理内存量,以字节为单位
/// </summary>
internal UInt64 ullTotalPhys;

/// <summary>
/// 当前可用的物理内存量,以字节为单位。这是可以立即重用而无需先将其内容写入磁盘的物理内存量。它是备用列表、空闲列表和零列表的大小之和
/// </summary>
internal UInt64 ullAvailPhys;

/// <summary>
/// 系统或当前进程的当前已提交内存限制,以字节为单位,以较小者为准。要获得系统范围的承诺内存限制,请调用GetPerformanceInfo
/// </summary>
internal UInt64 ullTotalPageFile;

/// <summary>
/// 当前进程可以提交的最大内存量,以字节为单位。该值等于或小于系统范围的可用提交值。要计算整个系统的可承诺值,调用GetPerformanceInfo核减价值CommitTotal从价值CommitLimit
/// </summary>

internal UInt64 ullAvailPageFile;

/// <summary>
/// 调用进程的虚拟地址空间的用户模式部分的大小,以字节为单位。该值取决于进程类型、处理器类型和操作系统的配置。例如,对于 x86 处理器上的大多数 32 位进程,此值约为 2 GB,对于在启用4 GB 调整的系统上运行的具有大地址感知能力的 32 位进程约为 3 GB 。
/// </summary>

internal UInt64 ullTotalVirtual;

/// <summary>
/// 当前在调用进程的虚拟地址空间的用户模式部分中未保留和未提交的内存量,以字节为单位
/// </summary>
internal UInt64 ullAvailVirtual;


/// <summary>
/// 预订的。该值始终为 0
/// </summary>
internal UInt64 ullAvailExtendedVirtual;

internal void Refresh()
{
dwLength = checked((UInt32)Marshal.SizeOf(typeof(MemoryStatusExE)));
}
}

定义引用库函数的入口:

public static partial class Native
{

/// <summary>
/// 检索有关系统当前使用物理和虚拟内存的信息
/// </summary>
/// <param name="lpBuffer"></param>
/// <returns></returns>
[LibraryImport("Kernel32.dll", SetLastError = true)]
[return: MarshalAs(UnmanagedType.Bool)]
internal static partial Boolean GlobalMemoryStatusEx(ref MemoryStatusExE lpBuffer);
}

然后调用 Kernel32.dll 中的函数:

public class Program
{
static void Main()
{
var result = GetValue();
Console.WriteLine($"当前实际可用内存量:{result.ullAvailPhys / 1000 / 1000}MB");
Console.ReadKey();
}

/// <exception cref="Win32Exception"></exception>
public static MemoryStatusExE GetValue()
{
var memoryStatusEx = new MemoryStatusExE();
// 重新初始化结构的大小
memoryStatusEx.Refresh();
// 刷新值
if (!Native.GlobalMemoryStatusEx(ref memoryStatusEx)) throw new Win32Exception("无法获得内存信息");
return memoryStatusEx;
}
}

使用 AOT 发布项目,执行 CsharpAot.exe 文件。

减少体积

在前面两个例子中可以看到 CsharpAot.exe 文件大约在 3MB 左右,但是这个文件还是太大了,那么我们如何进一步减少 AOT 文件的大小呢?

读者可以从这里了解如何裁剪程序:https://learn.microsoft.com/zh-cn/dotnet/core/deploying/trimming/trim-self-contained

需要注意的是,裁剪是没有那么简单的,里面配置繁多,有一些选项不能同时使用,每个选项又能带来什么样的效果,这些选项可能会让开发者用得很迷茫。

经过笔者的大量测试,笔者选用了以下一些配置,能够达到很好的裁剪效果,供读者测试。

首先,引入一个库:

<ItemGroup>
<PackageReference Include="Microsoft.DotNet.ILCompiler" Version="7.0.0-*" />
</ItemGroup>

接着,在项目文件中加入以下选项:

<!--AOT 相关-->
<PublishAot>true</PublishAot>
<TrimMode>full</TrimMode>
<RunAOTCompilation>True</RunAOTCompilation>
<PublishTrimmed>true</PublishTrimmed>
<TrimmerRemoveSymbols>true</TrimmerRemoveSymbols>
<PublishReadyToRunEmitSymbols>false</PublishReadyToRunEmitSymbols>
<DebuggerSupport>false</DebuggerSupport>
<EnableUnsafeUTF7Encoding>true</EnableUnsafeUTF7Encoding>
<InvariantGlobalization>true</InvariantGlobalization>
<HttpActivityPropagationSupport>false</HttpActivityPropagationSupport>
<MetadataUpdaterSupport>true</MetadataUpdaterSupport>
<UseSystemResourceKeys>true</UseSystemResourceKeys>
<IlcDisableReflection >true</IlcDisableReflection>

最后,发布项目。

吃惊!生成的可执行文件只有 1MB 了,而且还可以正常执行。

笔者注:虽然现在看起来 AOT 的文件很小了,但是如果使用到 HttpClientSystem.Text.Json 等库,哪怕只用到了一两个函数,最终包含这些库以及这些库使用到的依赖,生成的 AOT 文件会大得惊人。

所以,如果项目中使用到其他 nuget 包的时候,别想着生成的 AOT 能小多少!

C# 导出函数

这一步可以从时总的博客中学习更多:https://www.cnblogs.com/InCerry/p/CSharp-Dll-Export.html

PS:时总真的太强了。

在 C 语言中,导出一个函数的格式可以这样:

// MyCFuncs.h
#ifdef __cplusplus
extern "C" { // only need to export C interface if
// used by C++ source code
#endif

__declspec( dllimport ) void MyCFunc();
__declspec( dllimport ) void AnotherCFunc();

#ifdef __cplusplus
}
#endif

当代码编译之后,我们就可以通过引用生成的库文件,调用 MyCFuncAnotherCFunc 两个方法。

如果不导出的话,别的程序是无法调用库文件里面的函数。

因为 .NET 7 的 AOT 做了很多改进,因此,.NET 程序也可以导出函数了。

新建一个项目,名字就叫 CsharpExport 吧,我们接下来就在这里项目中编写我们的动态链接库。

添加一个 CsharpExport.cs 文件,内容如下:

using System.Runtime.InteropServices;

namespace CsharpExport
{
public class Export
{
[UnmanagedCallersOnly(EntryPoint = "Add")]
public static int Add(int a, int b)
{
return a + b;
}
}
}

然后在 .csproj 文件中,加上 PublishAot 选项。

然后通过以下命令发布项目,生成链接库:

 dotnet publish -p:NativeLib=Shared -r win-x64 -c Release

看起来还是比较大,为了继续裁剪体积,我们可以在 CsharpExport.csproj 中加入以下配置,以便生成更小的可执行文件。

<!--AOT 相关-->
<PublishAot>true</PublishAot>
<TrimMode>full</TrimMode>
<RunAOTCompilation>True</RunAOTCompilation>
<PublishTrimmed>true</PublishTrimmed>
<TrimmerRemoveSymbols>true</TrimmerRemoveSymbols>
<PublishReadyToRunEmitSymbols>false</PublishReadyToRunEmitSymbols>
<DebuggerSupport>false</DebuggerSupport>
<EnableUnsafeUTF7Encoding>true</EnableUnsafeUTF7Encoding>
<InvariantGlobalization>true</InvariantGlobalization>
<HttpActivityPropagationSupport>false</HttpActivityPropagationSupport>
<MetadataUpdaterSupport>true</MetadataUpdaterSupport>
<UseSystemResourceKeys>true</UseSystemResourceKeys>
<IlcDisableReflection >true</IlcDisableReflection>

C# 调用 C# 生成的 AOT

在本小节中,将使用 CsharpAot 项目调用 CsharpExport 生成的动态链接库。

CsharpExport.dll 复制到 CsharpAot 项目中,并配置 始终复制

CsharpAotNative 中加上:

 [LibraryImport("CsharpExport.dll", SetLastError = true)]
[return: MarshalAs(UnmanagedType.I4)]
internal static partial Int32 Add(Int32 a, Int32 b);

然后在代码中使用:

 static void Main()
{
var result = Native.Add(1, 2);
Console.WriteLine($"1 + 2 = {result}");
Console.ReadKey();
}

在 Visual Studio 里启动 Debug 调试:

可以看到,是正常运行的。

接着,将 CsharpAot 项目发布为 AOT 后,再次执行:

可以看到,.NET AOT 调用 .NET AOT 的代码是没有问题的。

Golang 部分

Go 生成 Windows 动态链接库,需要安装 GCC,通过 GCC 编译代码生成对应平台的文件。

安装 GCC

需要安装 GCC 10.3,如果 GCC 版本太新,会导致编译 Go 代码失败。

打开 tdm-gcc 官网,通过此工具安装 GCC,官网地址:

https://jmeubank.github.io/tdm-gcc/download/

下载后,根据提示安装。

然后添加环境变量:

D:\TDM-GCC-64\bin

运行 gcc -v,检查是否安装成功,以及版本是否正确。

Golang 导出函数

本节的知识点是 cgo,读者可以从这里了解更多:

https://www.programmerall.com/article/11511112290/

新建一个 Go 项目:

新建一个 main.go 文件,文件内容如下:

package main

import (
"fmt"
)
import "C"

//export Start
func Start(arg string) {
fmt.Println(arg)
}

// 没用处
func main() {
}

在 Golang 中,要导出此文件中的函数,需要加上 import "C",并且 import "C" 需要使用独立一行放置。

//export {函数名称} 表示要导出的函数,注意,//export 之间 没有空格。

main.go 编译为动态链接库:

go build -ldflags "-s -w" -o main.dll -buildmode=c-shared main.go

不得不说,Go 编译出的文件,确实比 .NET AOT 小一些。

前面,笔者演示了 .NET AOT 调用 .NET AOT ,那么, Go 调用 Go 是否可以呢?

答案是:不可以。

因为 Go 编译出来的 动态链接库本身带有 runtime,Go 调用 main.dll 时 ,会出现异常。

具有情况可以通过 Go 官方仓库的 Issue 了解:https://github.com/golang/go/issues/22192

这个时候,.NET 加 1 分。

虽然 Go 不能调用 Go 的,但是 Go 可以调用 .NET 的。在文章后面会介绍。

虽然说 Go 不能调用自己,这里还是继续补全代码,进一步演示一下。

Go 通过动态链接库调用函数的示例:

func main() {
maindll := syscall.NewLazyDLL("main.dll")
start := maindll.NewProc("Start")

var v string = "测试代码"
var ptr uintptr = uintptr(unsafe.Pointer(&v))
start.Call(ptr)
}

代码执行后会报错:

.NET C# 和 Golang 互调

C# 调用 Golang

main.dll 文件复制放到 CsharpAot 项目中,设置 始终复制

然后在 Native 中添加以下代码:

 [LibraryImport("main.dll", SetLastError = true)]
internal static partial void Start(IntPtr arg);

调用 main.dll 中的函数:

 static void Main()
{
string arg = "让 Go 跑起来";
// 将申请非托管内存string转换为指针
IntPtr concatPointer = Marshal.StringToHGlobalAnsi(arg);
Native.Start(concatPointer);
Console.ReadKey();
}

在 .NET 中 string 是引用类型,而在 Go 语言中 string 是值类型,这个代码执行后,会出现什么结果呢?

执行结果是输出一个长数字。

笔者不太了解 Golang 内部的原理,不确定这个数字是不是 .NET string 传递了指针地址,然后 Go 把指针地址当字符串打印出来了。

因为在 C、Go、.NET 等语言中,关于 char、string 的内部处理方式不一样,因此这里的传递方式导致了跟我们的预期结果不一样。

接着,我们将 main.go 文件的 Start 函数改成:

//export Start
func Start(a,b int) int{
return a+b
}

然后执行命令重新生成动态链接库:

go build -ldflags "-s -w" -o main.dll -buildmode=c-shared main.go

main.dll 文件 复制到 CsharpAot 项目中,将 Start 函数引用改成:

 [LibraryImport("main.dll", SetLastError = true)]
[return: MarshalAs(UnmanagedType.I4)]
internal static partial Int32 Start(int a, int b);

执行代码调用 Start 函数:

 static void Main()
{
var result = Native.Start(1, 2);
Console.WriteLine($"1 + 2 = {result}");
Console.ReadKey();
}

Golang 调用 C#

CsharpExport.dll 文件复制放到 Go 项目中。

main 的代码改成:

func main() {
maindll := syscall.NewLazyDLL("CsharpExport.dll")
start := maindll.NewProc("Add")

var a uintptr = uintptr(1)
var b uintptr = uintptr(2)
result, _, _ := start.Call(a, b)

fmt.Println(result)
}

将参数改成 19,再次执行:

其他

在本文中,笔者演示了 .NET AOT,虽然简单的示例看起来是正常的,体积也足够小,但是如果加入了实际业务中需要的代码,最终生成的 AOT 文件也是很大的。

例如,项目中使用 HttpClient 这个库,会发现里面加入了大量的依赖文件,导致生成的 AOT 文件很大。

在 .NET 的库中,很多时候设计了大量的重载,同一个代码有好几个变种方式,以及函数的调用链太长,这样会让生成的 AOT 文件变得比较臃肿。

目前来说, ASP.NET Core 还不支持 AOT,这也是一个问题。

在 C# 部分,演示了如果使用 C# 调用系统接口,这里读者可以了解一下 pinvoke:http://pinvoke.net/

这个库封装好了系统接口,开发者不需要自己撸一遍,通过这个库可以很轻松地调用系统接口,例如笔者最近在写 MAUI 项目,通过 Win32 API 控制桌面窗口,里面就使用到 pinvoke 简化了大量代码。

本文是笔者熬夜写的,比较赶,限于水平,文中可能会有错误的地方,望大佬不吝指教。


相关推荐

字节跳动官方出品的免费图标库,超好用还能自定义修改

现在很多企业公司或品牌都会将自己开发的设计素材开放出来,像是图标集、字体或是网页框架等等,使这些项目可以被更多人使用,或是在开源的情况下创造出更多可能性,当然一方面也能提高被看到的机会。之前介绍过IB...

微信底部不起眼的图标,还藏着设计师的这些小心思,网友:贴心

每天至少要打开十几次的微信,你注意过底部的小图标吗?可能有些读者会说了,不就是几个小图标吗?你发现没:选中的图标选项框是实心图标,但是没有选中的就是空心图标。为什么会这样设计呢?所以,我们今天就来聊聊...

ppt常用小图标去哪里找?3个矢量素材网站推荐!

ppt是一个注重可视化表达的演示载体,除了高清图片,ppt中另一类常用的素材是各种小图标,也叫矢量图标,巧妙运用小图标能提升整体美观度和表现力,那么ppt常用小图标去哪里找呢?...

手机右上角的状态栏,如果出现了这些小图标,记得一定要关闭

文|技术探索者编辑|技术探索者...

Win10设置应用更新:升级“关于”页面,主页图标可跟随主题色

IT之家12月24日消息,科技媒体WindowsLatest昨日(12月23日)发布博文,报道称Windows10系统正整合两项“新”功能/特性,包括色彩更丰富的设置H...

全网最炫酷、最华丽、最好看的动画图标库

很多小伙伴一直苦恼找不到合适自己使用的动画图标,今天TJ君要给大家分享一个...

电脑任务栏的网络小图标,就能告诉你不能上网的真正原因是什么?

我们在使用电脑时,经常遇到电脑突然就不能上网了,特别是着急上传文件或资料的时候,偏偏在关键时候掉链子。其实,电脑真的很智通,只是我们不在意电脑告诉我们的重要信息,所以就只能干着急了。1、正常有线网络连...

建议收藏的7个高质量图标网站,一网打尽图标素材

大家好,这里是@秋叶PPT图标是PPT制作中经常用到的元素之一,其比文字生动,比图片简洁的特性,应用在PPT中可以极大地提高页面的可阅读性,从而深受广大PPT爱好者喜爱。...

软网推荐:轻轻松松制作与处理图标

在制作软件、网页、多媒体作品时,常常会涉及到各种各样的图标。要制作这些图标,我们当然可以使用Photoshop这样的专业软件,但用该软件制作,操作不见得简单。其实,我们只需使用一个便携版的AniFX...

服务器安全:CentOS 7中安装CSF防火墙

提到服务器的安全性,有人可能会觉得一般服务器的操作系统相对个人PC的安全性更高,更难以破解,尤其是CentOS操作系统自身也比较安全,但是因为服务器内的信息更为关键,所以笔者建议还是不要让它“裸奔”,...

什么是基因突变,乳腺癌常见的基因突变有哪些?

基因突变当一个细胞生长和分裂时,它会复制它的DNA。有时新细胞中的DNA与原始细胞不同。这叫做突变。你生来就可能有遗传的,或遗传的,你从你的父母那里继承下来的突变。如果他们中的一个生来就有一个基因,他...

.NET 7 AOT 的使用以及 .NET 与 Go 互相调用

目录...

6 张图带你彻底搞懂分布式事务 XA 模式

XA协议是由X/Open组织提出的分布式事务处理规范,主要定义了事务管理器TM和局部资源管理器RM之间的接口。目前主流的数据库,比如oracle、DB2都是支持XA协议的。...

【云原生】Prometheus AlertManager讲解与实战操作

一、概述Prometheus包含一个...

Win10桌面/手机版最深层次开发功能挖掘

IT之家讯Win10开发者预览版为我们提供了一个Win10大框架的早期概览,使开发者与热心用户都可以提前感受Win10带来的新特性,尝试新工具,而作为开发者,最关心的莫过于Windows多平台通用应...

取消回复欢迎 发表评论: