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

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

yuyutoo 2025-01-12 19:59 2 浏览 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 简化了大量代码。

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


相关推荐

史上最全的浏览器兼容性问题和解决方案

微信ID:WEB_wysj(点击关注)◎◎◎◎◎◎◎◎◎一┳═┻︻▄(页底留言开放,欢迎来吐槽)●●●...

平面设计基础知识_平面设计基础知识实验收获与总结
平面设计基础知识_平面设计基础知识实验收获与总结

CSS构造颜色,背景与图像1.使用span更好的控制文本中局部区域的文本:文本;2.使用display属性提供区块转变:display:inline(是内联的...

2025-02-21 16:01 yuyutoo

写作排版简单三步就行-工具篇_作文排版模板

和我们工作中日常word排版内部交流不同,这篇教程介绍的写作排版主要是用于“微信公众号、头条号”网络展示。写作展现的是我的思考,排版是让写作在网格上更好地展现。在写作上花费时间是有累积复利优势的,在排...

写一个2048的游戏_2048小游戏功能实现

1.创建HTML文件1.打开一个文本编辑器,例如Notepad++、SublimeText、VisualStudioCode等。2.将以下HTML代码复制并粘贴到文本编辑器中:html...

今天你穿“短袖”了吗?青岛最高23℃!接下来几天气温更刺激……

  最近的天气暖和得让很多小伙伴们喊“热”!!!  昨天的气温到底升得有多高呢?你家有没有榜上有名?...

CSS不规则卡片,纯CSS制作优惠券样式,CSS实现锯齿样式

之前也有写过CSS优惠券样式《CSS3径向渐变实现优惠券波浪造型》,这次再来温习一遍,并且将更为详细的讲解,从布局到具体样式说明,最后定义CSS变量,自定义主题颜色。布局...

柠檬科技肖勃飞:大数据风控助力信用社会建设

...

你的自我界限够强大吗?_你的自我界限够强大吗英文

我的结果:A、该设立新的界限...

行内元素与块级元素,以及区别_行内元素和块级元素有什么区别?

行内元素与块级元素首先,CSS规范规定,每个元素都有display属性,确定该元素的类型,每个元素都有默认的display值,分别为块级(block)、行内(inline)。块级元素:(以下列举比较常...

让“成都速度”跑得潇潇洒洒,地上地下共享轨交繁华
让“成都速度”跑得潇潇洒洒,地上地下共享轨交繁华

去年的两会期间,习近平总书记在参加人大会议四川代表团审议时,对治蜀兴川提出了明确要求,指明了前行方向,并带来了“祝四川人民的生活越来越安逸”的美好祝福。又是一年...

2025-02-21 16:00 yuyutoo

今年国家综合性消防救援队伍计划招录消防员15000名

记者24日从应急管理部获悉,国家综合性消防救援队伍2023年消防员招录工作已正式启动。今年共计划招录消防员15000名,其中高校应届毕业生5000名、退役士兵5000名、社会青年5000名。本次招录的...

一起盘点最新 Chrome v133 的5大主流特性 ?

1.CSS的高级attr()方法CSSattr()函数是CSSLevel5中用于检索DOM元素的属性值并将其用于CSS属性值,类似于var()函数替换自定义属性值的方式。...

竞走团体世锦赛5月太仓举行 世界冠军杨家玉担任形象大使

style="text-align:center;"data-mce-style="text-align:...

学物理能做什么?_学物理能做什么 卢昌海

作者:曹则贤中国科学院物理研究所原标题:《物理学:ASourceofPowerforMan》在2006年中央电视台《对话》栏目的某期节目中,主持人问过我一个的问题:“学物理的人,如果日后不...

你不知道的关于这只眯眼兔的6个小秘密
你不知道的关于这只眯眼兔的6个小秘密

在你们忙着给熊本君做表情包的时候,要知道,最先在网络上引起轰动的可是这只脸上只有两条缝的兔子——兔斯基。今年,它更是迎来了自己的10岁生日。①关于德艺双馨“老艺...

2025-02-21 16:00 yuyutoo

取消回复欢迎 发表评论: