不安装运行时运行.NET程序
好久没写文章了,有些同学问我公众号是不是废了?其实并没有。其实想写的东西很多很多,主要是最近公司比较忙,以及一些其他个人原因没有时间来更新文章。这几天抽空写了一点点东西,证明公众号还活着。 长久以来的认知,对于托管代码 .NET / JAVA ,都是需要在服务器上安装 SDK 或者运行时的。比如 .NET Framework 4.XX ,JDK/JRE.XX 等。其实从 .NET Core 2.1 开始我们的 .NET 程序可以独立打包成可以执行文件,在服务器上根本不需要安装任何运行时相关的东西就可以运行。这个发布模式在某些情况下可以大大提高部署的效率。以下简单介绍一下。
“独立”部署模式
在发布界面部署模式选择“独立”,点击保存之后然后正常发布。等到发布完成之后,查看 publish 目录,可以发现里面生成了一大堆文件,数量有上百个。这里其实就包含了 runtime 相关的文件。
我们把这堆文件全部复制到某个未安装过 .NET SDK 或者 runtime 的 windows 服务器上,找到 SelfContainedTest.exe 文件,双击运行。如果一切顺利,会启动一个控制台。
访问一下服务器的 5000 口,看到测试数据被成功的输出了,证明我们的 .NET 程序可以正常运行了 。
单文件
上面的操作我们已经可以不安装运行时在服务器上运行 .NET 程序了。但是那么多文件看着不太优雅,下面让我们的 .NET 程序打包成一个文件。 打开发布设置界面,勾上“生成单个文件”
点击保存,发布之后,在 publish 目录可以看到只剩下 6 个文件了。排除配置文件,pdb 文件等,其实真正的程序只是 SelfContainedTest.exe 文件,所以称之为单文件。双击这个文件我们的程序就可以正常的运行了。
裁剪
以上我们已经把程序从多个文件打包成一个文件了。这个文件我们可以看到有 70 M ,对于我们一个简单的演示程序来说 70M 也挺大了。那么有什么办法来缩小我们的可以执行文件吗? 其实我们只要在发布配置上打开裁剪功能,就可以缩小我们的程序。
在发布配置界面勾上“裁剪未使用的代码”,点击保存,发布之后,在 publish 文件夹下面生成的 SelfContainedTest.exe 文件缩小到了 30M 左右。
裁剪的注意点
这里大概说一下裁剪的原理。当我们使用裁剪功能的时候,发布程序会开始分析我们的代码,哪些类被使用,哪些类没有使用,没有使用的类就会被删除掉,使用这样的原理来减小发布后程序集的大小。 但是以上方法显然会有一个问题,那就是无法识别动态性很强的代码,比如反射实现的某些功能。比如以下代码:
string s = Console.ReadLine();
Type type = Type.GetType(s);
foreach (var m in type.GetMethods())
{
Console.WriteLine(m.Name);
}
显然以上代码静态分析没办法知道程序最终需要使用那些类,因为目标类是通过 Console.ReadLine 方法输入进去的。在程序没有执行的时候谁也不知道哪些类会被使用。
在 IIS 上运行
上面我们演示程序运行的时候是寄宿在控制台上的,这样的话很容易被人误关闭。其实单文件发布的程序照样可以使用 IIS 来托管。 按照正常的 IIS 发布网站的流程配置之后,把应用程序池设置为 “无托管代码” 访问对应的端口程序就可以正常运行了。
在 linux 上运行
以上我们都是在 windows 上测试,现在让我们试一下在 linux 上运行它。 在 linux 上运行的话,需要在发布配置界面修改“目标运行时”为 linux-64 。
发布成功后把生成的文件复制到 linux 服务器上。cd 到目录,运行以下代码。
chmod +x SelfContainedTest
./SelfContainedTest
很不幸,我们的程序没有按计划运行起来。
通过搜索后发现,需要设置一个环境变量。修改运行的代码:
export DOTNET_SYSTEM_GLOBALIZATION_INVARIANT=1
./SelfContainedTest
运行完之后我们的程序应该会顺利的启动。访问一下对应的 http 接口,可以看到正确的输出了。
总结
通过以上演示,我们根本没必要在服务器(windows/linux)上安装任何 SDK 或者运行时就可以完整的运行我们的 .NET 程序。而且通过裁剪之后我们的程序的大小也缩小到了一个很小的范围。以上功能对于互联网行业来说可能没什么必要,毕竟大家走的都是容器化部署,服务器上本来就不需要安装运行时。但是对于一些传统行业,比如医院这样的环境,还有很多需要在服务器上人肉部署的场景。在这些场景之下就非常有意义了,可以大大的体高部署的效率。毕竟不是谁都可以很快的在服务器上安装好运行时,特别是 linux 服务器。 其实不安装运行时来运行程序还有一个办法,那就是使用 AOT 发布,这个我们下次再讲。
相关文章
- 在 .NET 6 中使用 dotnet format 格式化代码
- 高性能微服务网关.NETCore客户端Kong.Net开源发布
- .net 温故知新:【8】.NET 中的配置从xml转向json
- 【.net】使用EF的一点小问题
- .Net 7 的 R2R,Crossgen2是什么?
- 【愚公系列】2023年01月 .NET CORE工具案例-基于MediatR的CQRS模式
- 使用Docker快速部署一个Net项目
- GUN/Linux通用Glibc库是如何操控.Net 7的CLR
- MySQL Variables net_read_timeout 数据库 参数变量解释及正确配置使用
- 开启.NETMySQL的无缝连接!(.net链接mysql)
- .net利用MSSQL连接ASP.NET的办法探索(mssql 连接asp)
- Net访问Oracle数据库的简易方法(.net访问oracle)
- Net开发Oracle数据库新技术攻关挑战(.net开发oracle)
- Net结合MySQL实现强劲的数据库运算(.net 用mysql)
- 使用NET来连接MySQL数据库的简单方法(.net怎么连mysql)
- NET和MySQL的无缝搭配实现创新的技术路径(.net mysql支持)
- NET 与 MySQL 结合能实现优雅的数据事务处理(.net MySQL事物)
- 探索Oracle中NET位置的背后秘密(oracle中net位置)
- Redis无法安装NET组件(redis装不上net)
- googlesuggest下拉菜单实现代码(asp.net版本)
- asp.net文件路径之获得虚拟目录的网站的根目录
- Asp.net中时间格式化的6种方法详细总结
- ASP.NET实现LicenseKey输入功能的小例子
- .net实现webservice简单实例分享
- ASP.NET实现根据IP获取省市地址的方法
- .net解决spider多次和重复抓取的方案
- Asp.net中Response.Charset与Response.ContentEncoding区别示例分析
- asp.net中IDataParameter调用存储过程的实现方法