深入理解.NET Core的基元: deps.json, runtimeconfig.json, dll文件
原文链接: Deep-dive into .NET Core primitives: deps.json, runtimeconfig.json, and dll's
作者: Nate McMaster
C#编译器(The C# Compiler)
C#的编译器可以将cs文件转换为dll文件, 即程序集文件。程序集文件是一个便携的可执行格式文件, 借助.NET Core,它可以运行在Windows, MacOS和Linux系统中。
在Windows系统中, .NET Core的编译器文件csc.dll存放在以下目录中
C:\Program Files\dotnet\sdk\[.NET Core 版本号]\Roslyn\bincore
笔者使用了2.1.400版本,所以编译器存放目录是C:\Program Files\dotnet\sdk\2.1.400\Roslyn\bincore
.NET Core编译器文件csc.dll
也是一个.NET Core应用程序,所以你可以使用dotnet
命令直接执行编译器
C:\test>dotnet C:\Program Files\dotnet\sdk\2.1.400\Roslyn\bincore\csc.dll --help
下面我们尝试手动编译一个cs文件。
首先我们先创建一个Program.cs
文件,内容如下:
/* Program.cs */
class Program
{
static void Main(string[] args)
=> System.Console.WriteLine("Hello World!");
}
然后我们使用命令行命令将其编译
C:\test>dotnet "C:\Program Files\dotnet\sdk\2.1.400\Roslyn\bincore\csc.dll"
-reference:"C:\Program Files\dotnet\sdk\NuGetFallbackFolder\microsoft.netcore.app\2.1.0\ref\netcoreapp2.1\System.Runtime.dll"
-reference:"C:\Program Files\dotnet\sdk\NuGetFallbackFolder\microsoft.netcore.app\2.1.0\ref\netcoreapp2.1\System.Console.dll"
-out:Program.dll
Program.cs
参数说明
"C:\Program Files\dotnet\sdk\2.1.400\Roslyn\bincore\csc.dll"
是编译器所在的路径-reference
参数表示编译中需要引用的dll, 该参数可以指定多个dll , 例子中我们引用了System.Runtime.dll和System.Console.dll-out
参数表示编译生成的dll路径Program.cs
表示编译的源文件地址
Program.cs
编译成功, Program.dll
生成完毕。
runtimeconfig.json
对于.NET Core应用程序来说runtimeconfig.json
是不可或缺的。它是用来配置运行时的。
如果缺少了这个文件,运行dll文件的时候会产生以下异常。
C:\test>dotnet Program.dll
A fatal error was encountered. The library 'hostpolicy.dll' required to execute the application was not found in '........'
这句话的意思是.NET Core缺少指定组件来运行程序。
为了解决这个问题,我们可以添加一个Program.runtimeconfig.json
, 其内容如下
{
"runtimeOptions": {
"framework": {
"name": "Microsoft.NETCore.App",
"version": "2.1.2"
}
}
}
这里的配置dotnet命令将使用Microsoft.NETCore.App
作为共享框架(Shared Framework)。当dotnet命令运行的时候,它会去runtimeconfig.json中读取版本号,然后去C:\Program Files\dotnet\shared\[库名]\[版本号]
目录下,搜索对应的dll文件
现在我们重新运行上面的命令,结果如下:
C:\test>dotnet Program.dll
Hello world!
Hello World被正确输出了。
包(Package)
包(Package)是.NET中共享代码的一种方式。在.NET中,包的格式是nupkg, nupkg文件是一个ZIP压缩文件, 里面包含了.NET程序集和一个包含元数据的xml文件
。
在.NET中,最著名的包是JSON.NET, 又称Newtonsoft.Json.它提供了一个JSON序列化和反序列化的API。我们可以从NuGet.org中下载最新版本11.0.2的nupkg文件,并解压放置在我们当前的代码目录的packages\Newtownsoft.Json\11.0.2子目录下。
为了演示如何手动导入包来编译项目, 我们修改Program.cs
, 输出一个序列化之后的对象, 代码如下:
class Program
{
static void Main(string[] args)
=> System.Console.WriteLine(
Newtonsoft.Json.JsonConvert.SerializeObject(new { greeting = "Hello World!" }));
}
然后我们使用如下命令,编译Program.cs
C:\test>dotnet "C:\Program Files\dotnet\sdk\2.1.400\Roslyn\bincore\csc.dll"
-reference:"C:\Program Files\dotnet\sdk\NuGetFallbackFolder\microsoft.netcore.app\2.1.0\ref\netcoreapp2.1\System.Runtime.dll"
-reference:"C:\Program Files\dotnet\sdk\NuGetFallbackFolder\microsoft.netcore.app\2.1.0\ref\netcoreapp2.1\System.Console.dll"
-reference:"C:\Program Files\dotnet\sdk\NuGetFallbackFolder\microsoft.netcore.app\2.1.0\ref\netcoreapp2.1\System.Collections.dll"
-reference:.\packages\Newtonsoft.Json\11.0.2\lib\netstandard1.3\Newtonsoft.Json.dll
-out:Program.dll
Program.cs
这里为什么要引入
System.Collections.dll
呢?
原因是我们在代码中使用了匿名类型new { greeting = "Hello World!" }
, 对于匿名类型, C#编译器会为其生成一个.Equals
的方法, 这个方法调用了定义在System.Collections.dll
中 的System.Collections.Generic.EqualityComparer
方法
编译成功,但是会出现一些警告(Warning)
Program.cs(4,35): warning CS1701: Assuming assembly reference 'System.Runtime, Version=4.0.20.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a' used by 'Newtonsoft.Json' matches identity 'System.Runtime, Version=4.2.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a' of 'System.Runtime', you may need to supply runtime policy
这意味着.Newtonsoft.Json的作者创建Newtonsoft.Json.dll
时,是使用4.0.20.0的System.Runtime
程序集, 但是系统当前使用的System.Runtime
程序集是4.2.0.0版本的。编译器警告你4.0.20.0和4.2.0.0版本可以有很大的差异。不过幸运的是,这些差异都是向后兼容的(all backwards comptible), 所以Newtonsoft.Json.dll
可以正常工作。如果想去除这个警告,我们可以使用-nowarn:CS1701
C:\test>dotnet "C:\Program Files\dotnet\sdk\2.1.400\Roslyn\bincore\csc.dll"
-reference:"C:\Program Files\dotnet\sdk\NuGetFallbackFolder\microsoft.netcore.app\2.1.0\ref\netcoreapp2.1\System.Runtime.dll"
-reference:"C:\Program Files\dotnet\sdk\NuGetFallbackFolder\microsoft.netcore.app\2.1.0\ref\netcoreapp2.1\System.Console.dll"
-reference:"C:\Program Files\dotnet\sdk\NuGetFallbackFolder\microsoft.netcore.app\2.1.0\ref\netcoreapp2.1\System.Collections.dll"
-reference:.\packages\Newtonsoft.Json\11.0.2\lib\netstandard1.3\Newtonsoft.Json.dll
-nowarn:CS1701
-out:Program.dll
Program.cs
动态链接(Dynamic Link)
在上一步中,我们编译了一个引用了Newtonsoft.Json.dll
的.NET Core程序,在引用Newtonsoft.Json.dll
之前,代码可以正常运行,但是引用Newtownsoft.Json.dll
之后,程序运行失败。
C:\test> dotnet Program.dll
Unhandled Exception: System.IO.FileNotFoundException: Could not load file or assembly 'Newtonsoft.Json, Version=10.0.0.0, Culture=neutral, PublicKeyToken=30ad4fe6b2a6aeed'. The system cannot find the file specified.
.NET是一个动态链接的运行时。编译器会为Program.dll
程序集添加Newtonsoft.Json.dll
的引用,但是不会复制它的代码。.NET Core运行时期望在Program.dll
运行的时候,动态查找并加载一个Newtonsoft.Json.dll
文件。这一点对于System.Runtime.dll
, System.Console.dll
以及其他System.*
的程序集也是一样。
.NET Core可以配置查找Newtonsoft.Json.dll
文件的目录范围,这里我们先简单的将Newtownsoft.Json.dll
拷贝到与Program.dll
相同的目录中, 然后重新运行Program.dll
C:\test>copy .\packages\Newtonsoft.Json\11.0.2\lib\netstandard1.3\Newtonsoft.Json.dll Newtonsoft.Json.dll
C:\test>dotnet Program.dll
{"greeting":"Hello World!"}
我们预期的结果出现了。
注意:这里不需要拷贝System.Runtime.dll和System.Console.dll, 原因是他们存在于Microsoft.NETCore.App共享框架中,我们已经在runtimeconfig.json中配置过了。
deps.json
正如上一节所说的.NET Core可以配置查找动态链接程序集的位置。
这些位置包括:
- 应用程序根目录(这个不需要配置)
- 包缓存目录
- 优化过的的包缓存或者运行时包商店
- 服务索引
- 共享框架(配置在runtimeocnfig.json中)
deps.json
是一个记录.NET Core中依赖清单的文件。它可以用来配置动态链接的程序集。
deps.json
文件中定义了动态链接的依赖列表。通常这个文件在Visual Studio中是自动生成,而且在生产环境中也会非常的大。但是它确实是一个纯文本文件,所以我们可以使用任何编辑器编写它。
下面我们手动添加一个Program.deps.json
, 代码如下:
{
"runtimeTarget": {
"name": ".NETCoreApp,Version=v2.1"
},
"targets": {
".NETCoreApp,Version=v2.1": {
"Newtonsoft.Json/11.0.2": {
"runtime": {
"lib/netstandard1.3/Newtonsoft.Json.dll": {}
}
}
}
},
"libraries": {
"Newtonsoft.Json/11.0.2": {
"type": "package",
"serviceable": false,
"sha512": ""
}
}
}
现在我们删除之前拷贝过来的Newtonsoft.Json.dll
, 然后重新运行Program.dll
C:\test>del Newtonsoft.Json.dll
C:\test>dotnet Program.dll
Error:
An assembly specified in the application dependencies manifest (Program.deps.json) was not found:
package: 'Newtonsoft.Json', version: '11.0.2'
path: 'lib/netstandard1.3/Newtonsoft.Json.dll'
由此可见,尽管我们添加了deps.json
, .NET Core依然需要一些其他的信息来探测deps.json
中定义的动态程序集。
这里有3种方式来设置,你可以选中一行任意一种方式设置搜索动态链接库的目录路径,修改之后,{"greeting":"Hello World!"}
就会正常输出出来。
*.runtimeconfig.dev.json
这种一种方式是最佳的实现方式.我们可以添加一个Program.runtimeconfig.dev.json
,并在其中添加动态链接搜索目录字段additionalProbingPaths
{
"runtimeOptions": {
"additionalProbingPaths": [
"/Users/nmcmaster/code/packages/"
]
}
}
注解:这里的配置类似于Transformed Config, 如果指定当前的环境是dev,它就会读取Program.runtimeconfig.json, 并将Program.runtimeconfig.dev.json的内容覆盖进去
命令行
我们还是可以使用dotnet exec
命令并指定--additionalprobingpath
参数来配置检索的目录。
C:\test> dotnet exec --additionalprobingpath ./packages/ Program.dll
*.runtimeconfig.json
当然你也可以直接在*.runtimeconfig.json
中添加additionalProbingPaths
字段
{
"runtimeOptions": {
"framework": {
"name": "Microsoft.NETCore.App",
"version": "2.1.2"
},
"additionalProbingPaths": [
"./packages/"
]
}
}
参考文献
- Specs on runtimeconfig.json and deps.json
- Packages, metapackages and frameworks
- .NET Core application deployment
- The implementation of the shared framework lookup
深入理解.NET Core的基元: deps.json, runtimeconfig.json, dll文件的更多相关文章
- 深入理解.NET Core的基元(二) - 共享框架
原文:Deep-dive into .NET Core primitives, part 2: the shared framework 作者:Nate McMaster 译文:深入理解.NET Co ...
- 深入理解.NET Core的基元(二)
原文:Deep-dive into .NET Core primitives, part 2: the shared framework作者:Nate McMaster译文:深入理解.NET Core ...
- 深入理解.NET Core的基元(三) - 深入理解runtimeconfig.json
原文:Deep-dive into .NET Core primitives, part 3: runtimeconfig.json in depth 作者:Nate McMaster 译文:深入理解 ...
- 深入 .NET Core 基础 - 1:deps.json, runtimeconfig.json 以及 dll
深入 .NET Core 基础:deps.json, runtimeconfig.json 以及 dll 原文地址:https://natemcmaster.com/blog/2017/12/21/n ...
- 谈谈C#基元类型
首先看一下.NET 中的基元类型,如下表: C# Type | .NET Framework Type -------------| ---------------------- bool | Sys ...
- 【C#进阶系列】28 基元线程同步构造
多个线程同时访问共享数据时,线程同步能防止数据损坏.之所以要强调同时,是因为线程同步问题实际上就是计时问题. 不需要线程同步是最理想的情况,因为线程同步一般很繁琐,涉及到线程同步锁的获取和释放,容易遗 ...
- CLR:基元类型、引用类型和值类型
最新更新请访问: http://denghejun.github.io 前言 今天重新看了下关于CLR基元类型的东西,觉得还是有必要将其记录下来,毕竟这是理解CLR成功 之路上的重要一步,希望你也 ...
- 《CLR via C#》读书笔记--基元类型、引用类型和值类型
编程语言的基元类型 编译器直接支持的数据类型称为基元类型.基元类型直接映射到Framework类库中存在的类型.例如:C#中的int直接映射到System.Int32类型.下表给出了C#基元类型与对应 ...
- CLR via C#深解笔记三 - 基元类型、引用类型和值类型 | 类型和成员基础 | 常量和字段
编程语言的基元类型 某些数据类型如此常用,以至于许多编译器允许代码以简化的语法来操纵它们. System.Int32 a = new System.Int32(); // a = 0 a = 1 ...
随机推荐
- Vue+localstrong登录注册,并保持登录状态
在router.js中添加meta区分 比如登录注册页面,不需要登录即可进入,那么我们把meta中的isLogin标志设置为false { //登录 path: '/login', component ...
- 关于<超文本>定义
百度百科定义: 1, 超文本是用超链接的方法,将各种不同空间的文字信息组织在一起的网状文本.它更是一种用户界面范式,用以显示文本及与文本之间相关的内容.现时超文本普遍以电子文档方式存在,其中的文字包含 ...
- linux使用storcli64查看硬盘信息
使用storcli查看硬盘信息: rpm -ivh storcli--.noarch.rpm cd /opt/MegaRAID/storcli/ ./storcli64 /c0(零) show 链接: ...
- HFS 轻量化 的文件服务器
国外的工具 国内的工具
- 支付宝红包口令自动复制到剪贴板脚本js,安卓,IOS通用版
有客户找到涛舅舅,要求开发一个可以自动支付宝红包口令的js脚本,经过大量探索和优化,目前此脚本功能已经测试成功! 预期效果: 只要来访用户在当前网页的任意位置点击一下,支付宝红包口令即可复制到用户手机 ...
- MySQL基于左右值编码的树形数据库表结构设计
MySQL基于左右值编码的树形数据库表结构设计 在关系型数据库中设计树形的数据结构一直是一个十分考验开发者能力的,最常用的方案有主从表方案和继承关系(parent_id)方案.主从表方案的最大缺点 ...
- Bad Hair Day [POJ3250] [单调栈 或 二分+RMQ]
题意Farmer John的奶牛在风中凌乱了它们的发型……每只奶牛都有一个身高hi(1 ≤ hi ≤ 1,000,000,000),现在在这里有一排全部面向右方的奶牛,一共有N只(1 ≤ N ≤ 80 ...
- Spring boot 处理 error 的套路
Spring boot 处理 error 的基本流程: Controller -> 发生错误 -> BasicErrorController -> 根据 @RequestMappin ...
- SPARK-AM-TrackURL-UI-500
HTTP ERROR 500 Problem accessing /proxy/application_1538120222810_0072/. Reason: Connection refused ...
- 用js限制网页只能在微信内置浏览器或支付宝内置浏览器中打开
function is_weixinOrAli(){ var ua = navigator.userAgent.toLowerCase(); //判断浏览器的类型 if (ua.match(/Micr ...