大家好,先祝大家国庆快乐。不过大家看到这篇文章的时候估计已经过完国庆了 ?。
上一篇我们写了如何通过 SelfContained 模式发布程序(不安装运行时运行.NET程序)达到不需要在目标机器上安装 runtime 就可以运行 .NET 程序的目标。其实除了标准的 self-contained 微软还给我们带来了 Native AOT 发布模式。是的你没看错,通过该技术我们的 .NET 程序会直接编译为 Native 代码而不再是 IL ,程序运行的时候直接就是机器码,不再需要 JIT 编译。通过 AOT 技术,我们的程序启动会变的非常快并且使用更少的内存,并且运行的时候不需要在机器上安装任何运行时。
前阶段 .NET7 发布了第一个 RC 版本,标志着正式版的 AOT 马上会随 .NET7 发布而到来。所以趁着国庆赶紧体验一把。
现阶段 .NET7 还在RC,所以我们选择安装 SDK 7.0.100-rc.1.22431.12 ,操作系统是 WIN10 64位,开发工具是 VS2022 17.4.0 Preview 2.1 。正式版的 VS2022 是没办法选择目标框架 .NET7 的,但是其实可以手动改 csproj 文件,所以 VS2022 Preview 不是必须的。
我们新建一个控制台程序,目标框架选择 NET7 (如果使用正式版的 VS2022 没有办法选择 net7 ,可以直接编辑 csproj 文件),右键项目选择“编辑项目文件”,在 PropertyGroup 节点下添加 PublishAot :
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<OutputType>Exe</OutputType>
<TargetFramework>net7.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable>
<!--aot发布-->
<PublishAot>true</PublishAot>
</PropertyGroup>
</Project>
修改 main 方法:
Console.WriteLine("Hello, AOT!");
Console.Read();
使用 dotnet 命令进行发布:
dotnet publish -r win-x64 -c Release

AOT 发布相比正常发布会慢一点,等待发布成功后,我们可以到以下目录查看 bin\Release\net7.0\win-x64\publish :

我们可以看到生成的 exe 文件只有 3.48MB ,相比普通单文件发布加裁剪过后的程序小了不少。

我们把这个 exe 程序复制到一台没有安装 .net 环境的服务器上,顺利运行起来了。
上面我们测试了一下控制台程序的 AOT 发布,相对比较简单没有什么问题。下面让我们试试应用范围最为广泛的 ASP.NET CORE 项目 AOT 发布行不行。
新建一个 ASP.NET CORE WebApi 项目,目标框架选择 NET7 。同样的操作编辑 csproj 文件,添加 PublishAot 属性:
<Project Sdk="Microsoft.NET.Sdk.Web">
<PropertyGroup>
<TargetFramework>net7.0</TargetFramework>
<Nullable>enable</Nullable>
<ImplicitUsings>enable</ImplicitUsings>
<PublishAot>true</PublishAot>
</PropertyGroup>
</Project>
同样使用 dotnet cli 命令进行发布:
dotnet publish -r win-x64 -c Release
不同于上面控制台项目的发布,ASP.NET CORE 项目的 AOT 发布会出现很多警告信息,暂且忽略。

等到发布完成后,我们看到生成了一个 27MB 大小的 exe 文件。双击运行起来,不得不提一句,这个启动速度真的是肉眼可见的快,双击之后瞬间就启动了。这个就是 AOT 发布最大的优势了。

访问一下默认生成的那个 Action 方法:http://localhost:5000/WeatherForecast/ 成功的输出了天气信息。

以上通过简单的测试,ASP.NET CORE WebApi 项目顺利的跑起来了, 当然他只是一个简单的示例项目,我们生产的项目相比这些要复杂多了。经过更深入的测试,发现现阶段 ASP.NET CORE 进行 AOT 发布后有一个比较麻烦的问题,那就是 JSON 序列化。
以下代码是默认生成的 WeatherForecastController 的 GET 方法,这个方法是个标准的同步方法,进行 AOT 发布后序列化没有任何问题。
[HttpGet]
public WeatherForecast[] Get()
{
return Enumerable.Range(1, 5).Select(index => new WeatherForecast
{
Date = DateTime.Now.AddDays(index),
TemperatureC = Random.Shared.Next(-20, 55),
Summary = Summaries[Random.Shared.Next(Summaries.Length)]
})
.ToArray();
}
但是如果把代码改成异步,或者说的更直白一点的话,返回值是 Task<T> 类型就会出现问题。比如把上面的代码使用 Task.FromResult 改造一下,使返回值变成 Task<WeatherForecast[]>
[HttpGet]
public async Task<WeatherForecast[]> Get()
{
var arr = Enumerable.Range(1, 5).Select(index => new WeatherForecast
{
Date = DateTime.Now.AddDays(index),
TemperatureC = Random.Shared.Next(-20, 55),
Summary = Summaries[Random.Shared.Next(Summaries.Length)]
})
.ToArray();
var result = await Task.FromResult(arr);
return result;
}
改造的程序进行 AOT 发布后运行,访问对应的接口程序不会有任何报错,但是返回值是个空对象的json:
{}
尝试修复该问题,并没有特别的好办法,目前能够勉强使用的办法是使用 System.Text.Json source generator 模式进行序列化:
首先编写一个 WeatherForecastContext 类继承 JsonSerializerContext,并且标记为 partial。为啥要标记为 partial ?因为类的另外部分是 source generator 自动生成的。
[JsonSerializable(typeof(Task<WeatherForecast[]>))]
internal partial class WeatherForecastContext : JsonSerializerContext
{
}
第二步,在配置 services 的时候顺便把 WeatherForecastContext 配置进去。
builder.Services.AddControllers()
.AddJsonOptions(options => options.JsonSerializerOptions.AddContext<WeatherForecastContext>());
通过以上操作,再次 AOT 发布后运行程序,访问接口,数据是能正确的返回了。但是有一点小瑕疵是Task对象自身的属性也被序列化出来了。
{
"result": [
{
"date": "2022-10-08T19:14:26.1801524+08:00",
"temperatureC": 6,
"temperatureF": 42,
"summary": "Warm"
},
{
"date": "2022-10-09T19:14:26.1816645+08:00",
"temperatureC": -9,
"temperatureF": 16,
"summary": "Bracing"
},
{
"date": "2022-10-10T19:14:26.1816648+08:00",
"temperatureC": -1,
"temperatureF": 31,
"summary": "Sweltering"
},
{
"date": "2022-10-11T19:14:26.181665+08:00",
"temperatureC": -17,
"temperatureF": 2,
"summary": "Balmy"
},
{
"date": "2022-10-12T19:14:26.1816651+08:00",
"temperatureC": -16,
"temperatureF": 4,
"summary": "Freezing"
}
],
"asyncState": null,
"creationOptions": 0,
"exception": null,
"id": 1,
"isCanceled": false,
"isCompleted": true,
"isCompletedSuccessfully": true,
"isFaulted": false,
"status": 5
}
以上对控制台程序,web 程序进行了测试,接下来顺便对桌面 GUI 程序测试一下吧。

很遗憾,不管是 WINFROM 还是 WPF 程序,进行 AOT 发布的时候直接都会报错,提示不支持。
AOT 发布的程序会有一些限制,我们编写的时候需要注意:
以上是直接复制的英文文档(原文地址在文末),因为英文不是很好,不进行翻译了,怕误导大家。主要需要注意的就是 1,2 两点 ,关于动态加载类库跟动态生成代码的问题。我想序列化的问题大概也就是出在这里,因为传统的序列化需要大量的使用动态生成代码技术。
通过以上我们对 .NET 上最常用的几种程序进行了 Native AOT 发布的测试。总体来说控制台跟ASP.NET CORE 项目能用,WINFROM 跟 WPF 不能用。比较遗憾的有两个点:
另外来说说性能,有同学可能觉得 Native AOT 之后性能会有很大的提升,毕竟大家都迷信 Native 速度快嘛。但是经过大佬们的测试事实上 AOT 之后跟没有 AOT 的代码性能基本在伯仲之间,有些地方甚至不如非 Native 的代码。为什么?因为非 Native 代码可以进行运行时 JIT 啊,可以在运行时分析代码对热点代码进行二次 JIT 来提升性能,而 Native AOT 之后的代码做不到这点。
Native AOT Deployment
Try the new System.Text.Json source generator
AOT和单文件发布对程序性能的影响

总的来说,我对ruby还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用
作为我的Rails应用程序的一部分,我编写了一个小导入程序,它从我们的LDAP系统中吸取数据并将其塞入一个用户表中。不幸的是,与LDAP相关的代码在遍历我们的32K用户时泄漏了大量内存,我一直无法弄清楚如何解决这个问题。这个问题似乎在某种程度上与LDAP库有关,因为当我删除对LDAP内容的调用时,内存使用情况会很好地稳定下来。此外,不断增加的对象是Net::BER::BerIdentifiedString和Net::BER::BerIdentifiedArray,它们都是LDAP库的一部分。当我运行导入时,内存使用量最终达到超过1GB的峰值。如果问题存在,我需要找到一些方法来更正我的代
我需要在客户计算机上运行Ruby应用程序。通常需要几天才能完成(复制大备份文件)。问题是如果启用sleep,它会中断应用程序。否则,计算机将持续运行数周,直到我下次访问为止。有什么方法可以防止执行期间休眠并让Windows在执行后休眠吗?欢迎任何疯狂的想法;-) 最佳答案 Here建议使用SetThreadExecutionStateWinAPI函数,使应用程序能够通知系统它正在使用中,从而防止系统在应用程序运行时进入休眠状态或关闭显示。像这样的东西:require'Win32API'ES_AWAYMODE_REQUIRED=0x0
Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack
我想用ruby编写一个小的命令行实用程序并将其作为gem分发。我知道安装后,Guard、Sass和Thor等某些gem可以从命令行自行运行。为了让gem像二进制文件一样可用,我需要在我的gemspec中指定什么。 最佳答案 Gem::Specification.newdo|s|...s.executable='name_of_executable'...endhttp://docs.rubygems.org/read/chapter/20 关于ruby-在Ruby中编写命令行实用程序
在选择我想要运行操作的频率时,唯一的选项是“每天”、“每小时”和“每10分钟”。谢谢!我想为我的Rails3.1应用程序运行调度程序。 最佳答案 这不是一个优雅的解决方案,但您可以安排它每天运行,并在实际开始工作之前检查日期是否为当月的第一天。 关于ruby-如何每月在Heroku运行一次Scheduler插件?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/8692687/
exe应该在我打开页面时运行。异步进程需要运行。有什么方法可以在ruby中使用两个参数异步运行exe吗?我已经尝试过ruby命令-system()、exec()但它正在等待过程完成。我需要用参数启动exe,无需等待进程完成是否有任何rubygems会支持我的问题? 最佳答案 您可以使用Process.spawn和Process.wait2:pid=Process.spawn'your.exe','--option'#Later...pid,status=Process.wait2pid您的程序将作为解释器的子进程执行。除
我构建了两个需要相互通信和发送文件的Rails应用程序。例如,一个Rails应用程序会发送请求以查看其他应用程序数据库中的表。然后另一个应用程序将呈现该表的json并将其发回。我还希望一个应用程序将存储在其公共(public)目录中的文本文件发送到另一个应用程序的公共(public)目录。我从来没有做过这样的事情,所以我什至不知道从哪里开始。任何帮助,将不胜感激。谢谢! 最佳答案 无论Rails是什么,几乎所有Web应用程序都有您的要求,大多数现代Web应用程序都需要相互通信。但是有一个小小的理解需要你坚持下去,网站不应直接访问彼此
我尝试运行2.x应用程序。我使用rvm并为此应用程序设置其他版本的ruby:$rvmuseree-1.8.7-head我尝试运行服务器,然后出现很多错误:$script/serverNOTE:Gem.source_indexisdeprecated,useSpecification.Itwillberemovedonorafter2011-11-01.Gem.source_indexcalledfrom/Users/serg/rails_projects_terminal/work_proj/spohelp/config/../vendor/rails/railties/lib/r
刚入门rails,开始慢慢理解。有人可以解释或给我一些关于在application_controller中编码的好处或时间和原因的想法吗?有哪些用例。您如何为Rails应用程序使用应用程序Controller?我不想在那里放太多代码,因为据我了解,每个请求都会调用此Controller。这是真的? 最佳答案 ApplicationController实际上是您应用程序中的每个其他Controller都将从中继承的类(尽管这不是强制性的)。我同意不要用太多代码弄乱它并保持干净整洁的态度,尽管在某些情况下ApplicationContr