.NET 8.0 本机 AOT

        在软件开发领域,优化性能和简化效率仍然至关重要。.NET 平台二十年来不断创新,为开发人员提供了构建弹性且高效的软件解决方案的基础架构。

        与本机 AOT提前编译相结合,取得了显着的进步。本文深入研究.NET Native AOT,揭示它的工作原理、优点以及它的各种应用场景。

什么是 .NET 本机 AOT?

        .NET 本机提前 (AOT) 编译是 .NET 平台中的一项前沿进步。通过 AOT,C# 代码将在开发人员计算机上编译为本机代码。这与在运行时将代码编译为本机代码的传统方法形成鲜明对比。

下面的架构说明了这一点。.NET 传统编译涉及两个步骤:

        1、C# 编译生成包含中间语言 (IL)代码的 DLL 文件。这样的 DLL 称为.NET 程序集。
        2、执行 .NET 程序时,.NET 运行时(CLR 公共语言运行时)会加载 .NET 程序集。CLR 的子系统负责将 IL 代码编译为由 CPU 直接执行的本机代码。这个子系统被命名为JIT(Just-In-Time)编译器。它之所以得名是因为它仅在首次调用某个方法时才编译该方法的 IL 代码。


        另一方面,.NET Native AOT 编译由一个步骤组成。在开发人员的机器上将 C# 源代码编译为本机代码。此过程涉及将 C# 代码转换为 IL 代码,然后转换为 Native 代码,形成两步编译过程。但这是一个实现细节。这就是下面架构中 AOT .NET 程序集框呈灰色的原因。

.NET 本机 AOT 的优点
.NET 本机提前 (AOT) 编译带来了一系列优势:

        增强的性能:通过将代码预编译为本机机器指令,.NET Native AOT 显着缩短了启动时间并提高了应用程序的整体性能。运行时期间没有 JIT 编译开销意味着执行速度更快,从而提供更流畅的用户体验。
        简化部署: AOT 编译的应用程序通常会生成具有零或更少依赖项的独立可执行文件。这简化了部署过程,使您可以更轻松地跨各种平台和设备分发应用程序,而无需额外安装或运行时组件。
更小的应用程序大小:通过修剪掉不必要的代码,AOT 可以大大减小应用程序的大小。这不仅节省了存储空间,还优化了应用程序的内存占用,这在移动设备或物联网设备等资源受限的环境中尤其重要。
        增强的知识产权保护: AOT 编译将源代码转换为优化的机器代码,使逆向工程尝试更具挑战性。与可以轻松反编译为原始 C# 代码的 IL 代码相比,生成的本机代码更加混乱且难以破译。这增强了应用程序中嵌入的敏感算法、业务逻辑和专有方法的安全性。

.NET Native AOT 的缺点
AOT 带来的好处不可避免地伴随着某些缺点。他们来了:

        特定于平台的编译: .NET Native AOT 生成特定于平台的本机代码,针对特定的体系结构或操作系统进行定制。例如,与常规 .NET 程序集不同,使用 AOT 在 Windows 上生成的可执行文件无法在 Linux 上运行。
        不支持跨操作系统编译:例如,您无法从 Windows 机器编译 Linux 本机版本,反之亦然。
        部分支持Reflection: Reflection依赖于动态代码生成和运行时类型发现,这与AOT编译代码的预编译和静态性质相冲突。然而,我们将在本文末尾看到,通常的反射用法与 AOT 配合得很好。
        需要 AOT 兼容的依赖项: AOT 编译要求项目中使用的所有库和依赖项都与 AOT 兼容。依赖反射、运行时代码生成或其他动态行为的库可能与 AOT 不兼容,从而可能导致冲突或运行时错误。
        增加构建时间: AOT 编译涉及在构建过程中预先生成本机代码。此附加步骤可以显着增加构建时间,特别是对于具有大量代码库的大型项目或应用程序。
        需要 C++ 桌面开发工具: AOT 只能在安装了这些工具的情况下进行编译,这些工具在您的硬盘上最多可达 7GB。

.NET 本机 AOT 实际应用
        使用 Visual Studio 创建 .NET 本机 AOT Web 应用程序,启动 Visual Studio 2022 v17.8(或更高版本)并选择从模板ASP.NET Core Web API(本机 AOT)创建项目。

        该模板为我们生成一个 .NET 8 ASP.NET Core Web API 应用程序,我们可以看到 .csproj 文件包含<PublishAot>true</PublishAot>。

编译 .NET Web 应用程序
        如果我们编译应用程序,我们可以看到在 .dll 下生成了一个 DLL 文件.\bin\debug\net8.0。此 DLL 是包含 IL 代码的常规 .NET 程序集。我们可以使用 ILSpy 对其进行反编译,如下图所示:

执行.NET Web应用程序
        此时,没有发生AOT编译。C# 代码在不到一秒的时间内就被编译为 IL 代码。.NET Web 应用程序可以按原样执行:

使用 .NET Native AOT 编译 Web 应用程序
        要使用 .NET Native AOT 编译 Web 应用程序,您必须dotnet publish 在包管理器控制台中键入:

        这次这个小应用程序的编译花费了 11 秒。我们获得一个大小为 9MB 的可执行文件。该文件无法使用 ILSpy 反编译,因为它不是 .NET 程序集。该文件仅包含本机代码。还生成一个 70MB PDB 文件,将本机代码与源代码链接起来以进行调试。

        可执行文件可以在任何 Windows x64 系统上按原样部署和执行。部署可以更简单吗?我不这么认为!

.NET 本机 AOT 权衡
现在是评估选项的时候了:

        1、一方面,我们在不到一秒的时间内生成了一个可在所有平台上运行的 40KB DLL。但它需要预安装.NET 8.0。
        2、另一方面,我们在 11 秒内编译出一个 9MB 的可执行文件。它可以在任何 Windows x64 系统上运行,无需任何先决条件。此外,这个本机版本的启动速度更快。对于任何足够复杂的 Web 应用程序,我们都可以期待更多的性能提升。
        人们可能会认为 9MB 比 40KB 大得多!但安装 .NET 8.0 需要的磁盘空间远多于 9MB。仅目录C:\Program Files\dotnet\shared\Microsoft.NETCore.App\8.0.0  就有 70MB,而且还有更多东西需要安装。因此 AOT 导致:

3、紧凑的容器镜像在容器化部署设置中特别有用
4、由于图像尺寸较小,部署时间缩短


以下是本文中值得引用的评论:

        这是怎样的前沿进步?在 .NET 平台引入之前的半个世纪里,这种软件的生产方式已经存在。
这是我的答案:因为你同时拥有了两个世界。

        1、一方面,与 C/C++ 相比,现代语言和平台通过运行时带来了许多便利
        2、另一方面,原始的本机可执行文件无需在生产计算机上安装平台。
        暂停一下,你就会意识到这个单一可执行文件有多酷。这 9MB 不仅包含 Web 应用程序代码。但它包含一些 CLR 代码(垃圾收集、库加载……)、常用的 .NET 类型(字符串、整数……)和 API 代码(如WebApplication.CreateSlimBuilder()【WebApplication.CreateSlimBuilder Method (Microsoft.AspNetCore.Builder) | Microsoft Learn】和 JSON 序列化器代码)。

        下面的依赖关系图(通过 NDepend 获得【Visual Studio - Explore existing .net code architecture】)显示了此 Web 应用程序代码及其依赖关系,全部打包在 9MB 位中!

用反射破坏.NET Native AOT?
        我努力尝试用 Reflection 来破坏 .NET Native AOT,但失败了。这是一个好消息,AOT 支持很多 Reflection API。以下是与 AOT 配合使用的代码:

// Reflection usage A
System.Type type = new object().GetType();
System.Reflection.MethodInfo methodInfo = type.GetMethod("ToString");
string str = (string)methodInfo.Invoke("Walk the dog", BindingFlags.Default, null, new object?[0], null);
Console.WriteLine("Reflection usage A:" + str);// Reflection usage B
var listOfString = typeof(List<>).MakeGenericType(new Type[] { typeof(string) });
var list = Activator.CreateInstance(listOfString) as List<string>;
list.Add("hello");
PropertyInfo prop = listOfString.GetProperty("Count");
int count = (int)prop.GetMethod.Invoke(list, new object?[0]);
Console.WriteLine("Reflection usage B:" + count);// Reflection usage C
Assembly assembly = Assembly.GetExecutingAssembly();
Type[] types = assembly.GetTypes();
foreach (Type type1 in types.Take(5)) {Console.WriteLine("Reflection usage C:" + type1.FullName);
}// Reflection usage D
Type unknownType = Type.GetType("System.String");
ParameterExpression param = Expression.Parameter(unknownType, "x");
MethodInfo method = unknownType.GetMethod("ToLower", Type.EmptyTypes);
Expression call = Expression.Call(param, method);
var lambda = Expression.Lambda(call, param).Compile();
var result = lambda.DynamicInvoke("HELLO");
Console.WriteLine("Reflection usage D:" + result);

        我们dotnet publish有时会收到一些警告,但它确实有效!

.NET 8 对本机 AOT 的支持
支持这些

1、Middleware 中间件
2、Minimal APIs 最小 API
3、gRPC 远程过程调用
4、Kestrel HTTP Server
5、Authorization 授权
6、JWT Authentication 认证
7、CORS 跨域资源共享
8、HealthChecks 健康检查
9、OutputCaching 输出缓存
10、RequestDecompression 请求解压
11、ResponseCaching 响应缓存
12、ResponseCompression 响应压缩
13、StaticFiles 静态文件
14、WebSockets
15、ADO.NET
16、PostgreSQL
17、Dapper AOT
18、SQLite

尚不支持这些

1、ASP.NET Core MVC
2、WebAPI
3、SignalR
4、Blazor Server,
5、Razor Pages,
6、Session, Spa
7、Entity Framework Core


结论
        .NET Native AOT 代表了优化 .NET 应用程序的关键一步。这种编译为本机代码的过程可以最大限度地减少运行时对即时 (JIT) 编译的依赖,从而提高性能。由此带来的好处包括更快的执行速度、减少的部署开销以及提高可扩展性的潜力,使其成为提高 .NET 开发领域效率和性能的绝佳选择。

        借助 .NET 8,Native AOT 已经相当成熟,可以在生产中使用。我们当然可以希望微软能够继续改进AOT支持。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/231710.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

paddlehub 文本检测使用

PaddleHub负责模型的管理、获取和预训练模型的使用。 参考&#xff1a;https://github.com/PaddlePaddle/PaddleHub/tree/develop/modules/image/text_recognition/chinese_text_detection_db_server import paddlehub as hub import cv2 # from utils import cv_show import…

MIT线性代数笔记-第31讲-线性变换及对应矩阵

目录 31.线性变换及对应矩阵打赏 31.线性变换及对应矩阵 线性变换相当于是矩阵的抽象表示&#xff0c;每个线性变换都对应着一个矩阵 例&#xff1a; 考虑一个变换 T T T&#xff0c;使得平面上的一个向量投影为平面上的另一个向量&#xff0c;即 T : R 2 → R 2 T:R^2 \to R…

DS|图(连通与生成树)

题目一&#xff1a;DS图 -- 图的连通分量 题目描述&#xff1a; 输入无向图顶点信息和边信息&#xff0c;创建图的邻接矩阵存储结构&#xff0c;计算图的连通分量个数。 输入要求&#xff1a; 测试次数t 每组测试数据格式如下&#xff1a; 第一行&#xff1a;顶点数 顶点…

DZ-200系列中间继电器 板后不带底座 DZY-212X DC220V JOSEF约瑟

DZY-200系列中间继电器 系列型号&#xff1a; DZY-201中间继电器 DZY-222中间继电器 DZY-202中间继电器 DZY-203中间继电器 DZY-204中间继电器 DZY-205中间继电器 DZY-206中间继电器 DZY-207中间继电器 DZY-208中间继电器 DZY-209中间继电器 DZY-210中间继电器 DZY-211中间继电…

Java IDEA JUnit 单元测试

JUnit是一个开源的 Java 单元测试框架&#xff0c;它使得组织和运行测试代码变得非常简单&#xff0c;利用JUnit可以轻松地编写和执行单元测试&#xff0c;并且可以清楚地看到哪些测试成功&#xff0c;哪些失败 JUnit 还提供了生成测试报告的功能&#xff0c;报告不仅包含测试…

vue封装基础input组件(添加防抖功能)

先看一下效果&#xff1a; // 调用页面 <template><div><!-- v-model&#xff1a;伪双向绑定 --><my-input v-model"inputVal" label"姓名" type"textarea" /></div> </template><script> import…

​已解决java.lang.ArrayIndexOutOfBoundsException异常的正确解决方法,亲测有效!!!​

已解决java.lang.ArrayIndexOutOfBoundsException异常的正确解决方法&#xff0c;亲测有效&#xff01;&#xff01;&#xff01; 目录 报错问题 解决思路 解决方法 总结 Q1 - 报错问题 java.long.ArrayIndexOutOfBoundsException 是Java中的一个运行时异常​&#xff0c…

FreeRTOS——内存管理知识总结及实战

1 freeRTOS动态创建与静态创建 动态创建&#xff1a;从FreeRTOS 管理的内存堆中申请创建对象所需的内存&#xff0c;在对象删除后&#xff0c; 这块内存释放回FreeRTOS管理的内存堆中 静态创建&#xff1a;需用户提供各种内存空间&#xff0c;并且使用静态方式占用的内存空间一…

简单vlan划分和dhcp中继(Cisco Packet Tracer模拟)

文章目录 1. 前言2. 功能实现2.1. dhcp服务器接入2.2. 学校web服务器2.3. 设置学校dns服务器2.4. 设置线路冗余2.5. 配置ac。 1. 前言 在这里我们的计网作业是使用思科的Cisco Packet Tracer进行对校园网的简单规划&#xff0c;这里我对校园网进行了简单的规划&#xff0c;功能…

MySQL之CRUD、常见函数及union查询

文章目录 1.思维导图 1.思维导图

java: 写入数据到HBase

一、添加依赖 <dependency><groupId>org.apache.hadoop</groupId><artifactId>hadoop-client</artifactId><version>2.6.0</version></dependency><dependency><groupId>org.apache.hbase</groupId><art…

python+django超市进销存仓库管理系统s5264

本次设计任务是要设计一个超市进销存系统&#xff0c;通过这个系统能够满足超市进销存系统的管理及员工的超市进销存管理功能。系统的主要功能包括&#xff1a;首页、个人中心、员工管理、客户管理、供应商管理、承运商管理、仓库信息管理、商品类别管理、由管理员和员工&#…

金和OA C6 MailTemplates.aspx sql注入漏洞

产品介绍 金和网络是专业信息化服务商,为城市监管部门提供了互联网监管解决方案,为企事业单位提供组织协同OA系统开发平台,电子政务一体化平台,智慧电商平台等服务。 漏洞概述 金和 OA C6 MailTemplates.aspx接口处存在SQL注入漏洞&#xff0c;攻击者除了可以利用 SQL 注入漏…

ES集群分片数据的高可用

文章目录 ES集群分片数据的高可用1. 集群设置索引节点2. 集群新增文档数据3.查看集群中文档数据分片节点4. 让节点9201宕机&#xff0c;查看其分片变化5. 让节点9201&#xff0c;查看分片变化 ES集群分片数据的高可用 集群中的索引主分片和副分片在不同的计算机上&#xff0c;如…

CSS新增文本描边-text-stroke属性

-webkit-text-stroke属性 概念&#xff1a;-webkit-text-stroke属性为文本添加描边效果。所谓的描边效果&#xff0c;指的是给文字添加边框 语法&#xff1a; -webkit-text-stroke:width color;Chrome和Firefox这两个浏览器都只能识别带有-webkit前缀的text-stroke属性 -web…

astadmin安装querylist插件Puppeteer

我本来是想在linux服务器上安装&#xff0c;折腾了一天也没安装成功&#xff0c;由于急着用&#xff0c;就先做window10上安装了&#xff0c;以后有时间再研究centos7上安装 一 首先需要安装fastadmin 框架和querylist插件 这个大家可以自行安装&#xff0c;querylist安装地址…

Django 7 实现Web便签

一、效果图 二、会用到的知识 目录结构与URL路由注册request与response对象模板基础与模板继承ORM查询后台管理 三、实现步骤 1. terminal 输入 django-admin startapp the_10回车 2. 注册&#xff0c; 在 tutorial子文件夹settings.py INSTALLED_APPS 中括号添加 "the…

Java-变量类型-分类

1 需求 2 接口 3 示例 public class RunoobTest {// 成员变量private int instanceVar;// 静态变量private static int staticVar;public void method(int paramVar) {// 局部变量int localVar 10;// 使用变量instanceVar localVar;staticVar paramVar;System.out.println(…

ubuntu 20.04 自由切换 python 的版本

问题描述 当前 ubuntu 20.04 默认安装了多个 python 的版本&#xff0c;执行 python 时&#xff0c;默认版本是 Python 2.7.18 zhangszzhangsz:~$ python Python 2.7.18 (default, Jul 1 2022, 12:27:04) [GCC 9.4.0] on linux2 Type "help", "copyright&quo…

Unity 圆角 线段 绘制 LineRender

需求 绘制圆角 核心函数 /// <summary>/// 点ABC 形成的角度必须为90 点c为中间的点/// </summary>/// <param name"a"></param>/// <param name"b"></param>/// <param name"c"></param>/// &…