我发现他们在文档的前两章中定义语言的方法和方式特别有趣。所以我决定试一试,从“你好,世界!”开始。

顺便说一句,我是在Windows 7 x64上这么做的。

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

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

我的期望是系统编程语言会产生一些紧凑的东西。

谁能详细解释一下Rust编译的目的是什么,它是如何从一个3行程序生成如此巨大的图像的?它是否编译到虚拟机?是否有我错过的strip命令(发布版本中的调试信息?)?还有什么能让我们理解的吗?


当前回答

默认情况下,Rust编译器会优化执行速度、编译速度和调试便捷性(例如,通过包含符号),而不是最小化二进制大小。

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

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

使用Rust 1.32.0或更新版本(默认不包括jemalloc) 在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。

使用货物构建-释放构建在释放模式

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

你也可以使用#![no_std]删除Rust的libstd。详见min-size -rust。

其他回答

这是一个功能,不是bug!

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

嘿,现在不是1978年了,很多人的电脑内存都超过了2mb:-)

默认情况下,Rust编译器会优化执行速度、编译速度和调试便捷性(例如,通过包含符号),而不是最小化二进制大小。

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

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

使用Rust 1.32.0或更新版本(默认不包括jemalloc) 在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。

使用货物构建-释放构建在释放模式

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

你也可以使用#![no_std]删除Rust的libstd。详见min-size -rust。

#![no_main]
#![no_std]

#[link(name = "msvcrt", kind = "dylib")]
extern {
    fn puts(ptr: *const u8); // i8 or u8 doesn't matter in this case
}

#[no_mangle]
unsafe extern fn main() {
    puts("Hello, World!\0".as_ptr());
}

#[panic_handler]
fn panic(_info: &core::panic::PanicInfo) -> ! {
    loop {}
}

下一个配置文件

[profile.release]
debug = false
strip = true
opt-level = 'z'
codegen-units = 1
lto = true
panic = 'abort'

用-r给出9 kb,而C

#include <stdio.h>

main() {
    puts("Hello, World!");
}

GCC和-Os给出48 kb, TCC给出2 kb。令人印象深刻,不是吗?

每晚安装防锈装置 rust工具链每晚安装,rust默认每晚安装

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

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

在底部,或者在[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'构建货物——发布

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

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

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

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

至少在Linux上,你还可以使用strip命令剥离二进制符号:

strip target/release/<binary>

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