“ 本文 最初发布于 Mark Rendle 的博客,经原作者授权由 InfoQ 中文站翻译并分享。”
笔者在Linux上安装了.Net Core runtime,并对新的CLI、CoreRT、ASP.Net Core进行了简单的测试与分析。
在之前 文章 中,我终于将整个.Net Core runtime和工具都安装到我工作用的Linux笔记本了,包括了:
(根据 命名声明 ,从现在起我将会把 ASP.NET 5 写成 ASP.NET Core )
昨晚,我开始尝试在CoreCLR/CoreFx上运行各种当前能在Mono runtime上运行的ASP.NET Core项目。 到目前为止它们都能工作,虽然我确实将那些我认为有问题的项目留到了最后,尤其是那些使用了Azure Storage SDK中较边缘部分的项目。:)
我在台式机上尝试通过那些能同时在CoreCLR和Mono的Docker容器中运行的各种服务来执行一些负载测试,我将会在这里公开那些实验结果。
我过去的两天里挖遍了 GitHub上dotnet组织 的各个仓库之后,我所见到的让我感到很兴奋。当然,我说的不仅仅是在Linux上运行。
新的 dotnet
CLI 与我大约在去年使用过的 dnx
, dnu
和 dnvm
命令有很大差别。那些都是shell脚本(或Bash函数);如果你执行一下 cat `which dnx`
就能看见脚本,它用Mono运行一个.Net DLL。
但如果你执行 cat `which dotnet`
你将会得到满屏的乱码,因为 dotnet
是一个本机执行文件。
我不知道现在那边具体发展成怎样了。在 dotnet-nightly
文件夹里翻一下,有很多大小相近的本机执行文件,这挺怪异的,或许它们只是.Net程序集的装载器,但也挺酷的。
由于好奇让这一切成为可能的基础,我进入了 CoreRT 仓库,里面对CoreRT的描述是这样的
一个为AOT(ahead of time compilation)场景优化的.NET Core runtime,附带.NET Native编译器工具链。
然而什么是工具链呢?
我找到了隐藏在文档目录中的 intro-to-corert.md 文件,其中解析了具体的情况。
这个项目的开发者(我想应该包含所有在微软内外的贡献者)正在构建一套工具,可以将MSIL byte-code(由Roslyn编译C#代码产生)预先编译成本机x86/64机器编码。
最初默认的实现使用新的64-bit CLR JIT编译器(去年夏天发布)RyuJIT来产生机器编码,但工具链还可以使用其他编译器,包括它们自己引用的 IL-to-C++ compiler (难以置信的小巧) and LLILC ,这个是建基于 LLVM 的CoreCLR中正在使用的JIT编译器,而且将来打算支持AOT编译。
.NET CLR包含一个Just-In-Time (JIT)编译器负责把MSIL bytecode转换成本机机器编码。它在你应用程序运行的时候进行转换,而且它是在各方法第一次被调用的时候才进行转换的;因此,“just in time”。这就是为什么同一个程序集能够在不同的CPU和操作系统上使用。
JIT编译器,一般来说,是优化成尽可能快地完成编译,而不是产生尽可能好的机器编码。
一个Ahead-Of-Time (AOT) 编译器,能为特定的目标CPU和操作系统把所有MSIL bytecode预先转换成本机机器编码,所以你无需单独的runtime就能够分发应用或程序库。
因为AOT编译器不像JIT编译器那样有微观时间的限制,在大部分情况下能够产生更高效、优化度高的机器编码。
CoreRT自身以CoreCLR和程序库的修改版本呈现,用不同的方式进行组织,对依赖项也进行过清理。我猜测至少有些部分开启了更高效的 无用代码删除(dead code elimination) ,使其二进制文件尽可能的小。
所有这些意味着C#即将进入 Go 的领地 - 一个跨平台、本地编译、带垃圾回收的编程语言。除此之外,C#当然拥有现代语言的特点,比如泛型、async/await等等。
在同一个页面, Roadmap 的下面,有这样的描述:
开始,我们的目标是本机可执行(又称“控制台程序”)。之后,我们将会把它扩展到包括ASP.NET 5程序...
而且除了当前ASP.NET Core网站应用的“XCOPY部署”模型 - 简单地转存一个目录树到服务器外,我们还可以期待“COPY部署”模型 - 仅转存一个单独的可执行文件。就如同一页面所述,这会导致Dockerfile看上去像下面这样:
FROM debian:jessie EXPOSE 5000 ADD mycompiledapp.exe / # The .exe is just there to make a point :) ENTRYPOINT /mycompiledapp
还不清楚这工具链是否能静态链接非.NET程序库(例如 libuv 或者 libcurl)到本机二进制文件,还是需要以动态链接 .dll
或 .so
文件的形式一并安装。两种形式,都将制成十分小巧的Docker映像,而且容器的启动也会非常快,不过,这一连串的想法很快引起了对 unikernels 的期待,.NET Native将来会否支持呢?
初步的答案,至少看上去是 “会。会支持的。”
这些项目明显都在初级阶段,所以没必要太早地兴奋过头,免得将来在roadmap上有些东西会 被残忍地杀掉 ,但总而言之,是时候成为一个C#开发者了。
致谢:我是受 Tugberk Ugurlu最近关于这方面的部落文章 激发才跳入这个兔子窝的。
原文链接: https://blog.rendle.io/what-ive-learned-about-dotnet-native
编后语《他山之石》是 InfoQ 中文站新推出的一个专栏,精选来自国内外技术社区和个人博客上的技术文章,让更多的读者朋友受益,本栏目转载的内容都经过原作者授权。文章推荐可以发送邮件到 editors@cn.infoq.com。