48

我可以在 MS Docs 中看到如何更改 C# 编译的文件对齐方式(通过项目设置和命令行)。

我在 Google 上搜索并看到文章解释说 512 字节的文件对齐会减小 .dll 的大小。我已经用不同的文件对齐方式测试了自己,并且看到了,是的,确实如此。

我的问题是:

为什么我要使用不同的文件对齐方式?必须有需要这样做的场景,否则就没有选项?

另外,它具体是做什么的?MSDN 页面谈论部分?什么是部分?

https://docs.microsoft.com/en-us/dotnet/csharp/language-reference/compiler-options/advanced#filealignment

4

1 回答 1

77

这是一个相当技术性的实现细节。要开始使用,您首先必须了解 PE32 文件的结构,即 Windows 中 DLL 和 EXE 的文件格式。对此的规范文章是 Matt Pietrik 的“Peering Inside the PE, A Tour of the Win32 Portable Executable File Format”。写于 17 年前,但仍然相关且可用

/filealign 设置是指 IMAGE_OPTIONAL_HEADER.FileAlignment 字段的值。它确定如何对齐部分中的原始数据。节是文件中的一段代码或数据。几乎完全是纯 .NET 程序集的数据。

文件格式和磁盘之间有着非常密切的关系。可执行映像用作 Windows 中内存映射文件的支持文件。通过将文件映射到虚拟内存地址空间来加载可执行文件。非常有效,加载 DLL 只涉及创建该映射,没有从文件中读取实际数据。当进程试图从一个节中读取一个字节时,这种情况会以一种惰性的方式发生。如果它还没有加载到内存中,则会产生页面错误,并且操作系统会从文件中读取 4096 个字节到内存中。最大的优势是您无需为不使用的数据或代码付费。也是第一次阅读时阅读[属性]很昂贵的原因。

文件对齐的相关性是部分中的原始数据如何排列。大多数包含机器代码的现代可执行文件使用 4096 字节的对齐方式,即虚拟内存页面的大小。这与包含托管代码的程序集不太相关,IL 只是数据。使用较小的对齐方式会很有意义,这样会浪费更少的空间。512 字节(不是千字节)是一个快乐的数字,它是 PE32 格式中允许的最小值。

我能想到的将选项添加到 UI 的唯一可能原因是,与其他编译器相比,C# 编译器的编译选项非常少。“其他”是生成本机代码的编译器。所以有这个选项是因为编译器有这个选项。[attributes] 涵盖了许多调整,很好地使编译器命令行简洁明了。但是文件对齐没有属性,需要在生成文件之前知道,一个属性就来不及了。

相反的例子是 C++ 编译器和链接器,IDE 提供了19 个属性页来设置它们。但仍然没有涵盖所有内容,真正晦涩的必须在“命令行”选项页面中设置。

于 2011-03-05T20:25:48.537 回答