247

刚刚找到 Rust 并阅读了文档的前两章,我发现他们定义语言的方法和方式特别有趣。所以我决定弄湿我的手指并从Hello world开始......

我在 Windows 7 x64 上这样做了,顺便说一句。

fn main() {
    println!("Hello, world!");
}

发布cargo build并查看结果,targets\debug我发现结果.exe为 3MB。经过一番搜索(货物命令行标志的文档很难找到......)我找到--release了选项并创建了发布版本。令我惊讶的是,.exe 的大小只变小了一点点:2.99MB 而不是 3MB。

所以,承认我是 Rust 及其生态系统的新手,我的期望是系统编程语言会产生一些紧凑的东西。

谁能详细说明 Rust 正在编译什么,它怎么可能从 3 班轮程序中生成如此巨大的图像?是编译成虚拟机吗?是否有我错过的条命令(发布版本中的调试信息?)?还有什么可以让我们了解正在发生的事情吗?

4

6 回答 6

213

Rust 使用静态链接来编译它的程序,这意味着即使是最简单的Hello world!程序所需的所有库都将被编译到您的可执行文件中。这也包括 Rust 运行时。

要强制 Rust 动态链接程序,请使用命令行参数-C prefer-dynamic;这将导致文件大小更小,但也需要 Rust 库(包括其运行时)在运行时可供您的程序使用。这实质上意味着如果计算机没有它们,您将需要提供它们,占用的空间比原来的静态链接程序占用的空间多。

出于可移植性的考虑,我建议您以静态链接 Rust 库和运行时的方式将您的程序分发给其他人。

于 2015-03-12T11:18:07.797 回答
133

有关减小 Rust 二进制文件大小的所有方法的概述,请参阅min-sized-rust存储库。

当前减少二进制大小的高级步骤是:

  1. jemalloc使用 Rust 1.32.0 或更新版本(默认不包括)
  2. 将以下内容添加到Cargo.toml
[profile.release]
opt-level = 'z'     # Optimize for size.
lto = true          # Enable Link Time Optimization
codegen-units = 1   # Reduce number of codegen units to increase optimizations.
panic = 'abort'     # Abort on panic
strip = true        # Strip symbols from binary*

*strip = true需要 Rust 1.59+。在较旧的 Rust 版本strip上,在生成的二进制文件上手动运行。

  1. 在发布模式下使用cargo build --release

使用 Rust 可以完成更多工作nightly,但我将保留这些信息,min-sized-rust因为由于使用了不稳定的特性,它会随着时间的推移而变化。

您也可以使用#![no_std]删除 Rust 的libstd. 详情请参阅min-sized-rust

于 2019-02-23T13:24:05.097 回答
91

我没有任何 Windows 系统可供尝试,但在 Linux 上,静态编译的 Rust hello world 实际上比等效的 C 小。如果您看到大小差异很大,可能是因为您正在链接 Rust 可执行文件静态和 C 一动态。

使用动态链接,您还需要考虑所有动态库的大小,而不仅仅是可执行文件。

所以,如果你想比较苹果和苹果,你需要确保两者都是动态的或者都是静态的。不同的编译器会有不同的默认值,所以你不能仅仅依靠编译器的默认值来产生相同的结果。

如果你有兴趣,这里是我的结果:

-rw-r--r-- 1 aij aij 63 Apr 5 14:26 printf.c
-rwxr-xr-x 1 aij aij 6696 Apr 5 14:27 printf.dyn
-rwxr-xr-x 1 aij aij 829344 Apr 5 14:27 printf.static
-rw-r--r-- 1 aij aij 59 Apr 5 14:26 puts.c
-rwxr-xr-x 1 aij aij 6696 Apr 5 14:27 puts.dyn
-rwxr-xr-x 1 aij aij 829344 Apr 5 14:27 puts.static
-rwxr-xr-x 1 aij aij 8712 Apr 5 14:28 rust.dyn
-rw-r--r-- 1 aij aij 46 Apr 5 14:09 rust.rs
-rwxr-xr-x 1 aij aij 661496 Apr 5 14:28 rust.static

这些是使用 gcc (Debian 4.9.2-10) 4.9.2 和 rustc 1.0.0-nightly (d17d6e7f1 2015-04-02) (built 2015-04-03) 编译的,都带有默认选项和-staticfor gcc 和-C prefer-dynamicfor锈迹斑斑

我有两个版本的 C hello world,因为我认为使用puts()可能会链接更少的编译单元。

如果你想尝试在 Windows 上重现它,这里是我使用的来源:

printf.c:

#include <stdio.h>
int main() {
  printf("Hello, world!\n");
}

放.c:

#include <stdio.h>
int main() {
  puts("Hello, world!");
}

锈.rs

fn main() {
    println!("Hello, world!");
}

另外,请记住,不同数量的调试信息或不同的优化级别也会产生影响。但我预计,如果您看到巨大的差异,那是由于静态链接与动态链接。

于 2015-04-05T19:42:15.043 回答
47

使用 Cargo 编译时,可以使用动态链接:

cargo rustc --release -- -C prefer-dynamic

这将大大减少二进制文件的大小,因为它现在是动态链接的。

至少在 Linux 上,您还可以使用以下strip命令剥离符号的二进制文件:

strip target/release/<binary>

这将使大多数二进制文件的大小大约减半。

于 2016-01-30T19:28:08.727 回答
1

每晚安装rust - rustup toolchain install nightly,rustup default nightly

现在,在项目中的所有Cargo.toml 文件中进行这些更改。

在 Cargo.toml 顶部添加cargo-features = ["strip"]before[package]

在底部,或之间[dependencies][package]添加,

[profile.release]
# strip = true  # Automatically strip symbols from the binary.
opt-level = "z"  # Optimize for size.
lto = true  # Enable link time optimization
codegen-units = 1  # Reduce parallel code generation units

现在用RUSTFLAGS='-C link-arg=-s' cargo build --release

我发现这些链接很有用 - https://collabora.com/news-and-blog/blog/2020/04/28/reducing-size-rust-gstreamer-plugin/https://github.com/johnthagen/min -sized-rusthttps://arusahni.net/blog/2020/03/optimizing-rust-binary-size.html

于 2021-09-24T20:10:59.737 回答
-12

这是一个功能,而不是一个错误!

您可以指定程序中使用的库版本(在项目关联的 Cargo.toml 文件中)(甚至是隐含的),以确保库版本兼容性。另一方面,这需要将特定库静态链接到可执行文件,从而生成大型运行时映像。

嘿,现在已经不是 1978 年了——很多人的电脑内存超过 2 MB :-)

于 2018-02-15T05:03:55.350 回答