发布 Rust 1.87.0 —— 以及 Rust 十周年!

原文链接

2025 年 5 月 15 日 · Rust 发布团队

在荷兰乌得勒支举办的 "10 Years of Rust" 庆典现场,Rust 团队高兴地宣布:Rust 1.87.0 正式发布!

今天的发布日恰逢 Rust 1.0 发布整整十周年!

感谢过去和现在无数为 Rust 做出贡献的伙伴们。愿 Rust 再迎来几十年的辉煌!🎉

按照自 1.0 以来从未间断的六周固定节奏,本次稳定版包含过去 beta 分支的全部改动。

如果你已经通过 rustup 安装了旧版 Rust,只需执行:

ruby 复制代码
$ rustup update stable

即可升级到 1.87.0。 如未安装,可在官网的 appropriate page on our website 获取 rustup,并阅读 detailed release notes for 1.87.0。

如果想提前帮忙测试未来版本,可以将默认工具链切到 beta(rustup default beta)或 nightly(rustup default nightly)。发现问题请随时反馈!

1.87.0 稳定版亮点

匿名管道(Anonymous pipes)

1.87 为标准库加入了匿名管道接口,并与 std::process::Command 的输入/输出方法打通。将 stdout 与 stderr 合流,如下所示就变得相当简单,而以前要么用额外线程,要么依赖平台专用函数。

rust 复制代码
use std::process::Command;
use std::io::Read;

let (mut recv, send) = std::io::pipe()?;

let mut command = Command::new("path/to/bin")
    .stdout(send.try_clone()?)  // stdout
    .stderr(send)               // stderr 共用同一管道
    .spawn()?;

let mut output = Vec::new();
recv.read_to_end(&mut output)?;

// 必须在子进程退出前读取完数据,否则若输出过大可能填满 OS 缓冲区。
assert!(command.wait()?.success());

安全调用架构内建函数(Safe architecture intrinsics) 此前因"需要目标 CPU 特性"而被标记为 unsafe 的大部分 std::arch 内建函数,如今在启用了相应特性的代码中可 直接以安全代码调用。下面这段用 AVX2 手写数组求和的示例,核心循环已无需 unsafe:

rust 复制代码
#![forbid(unsafe_op_in_unsafe_fn)]

use std::arch::x86_64::*;

fn sum(slice: &[u32]) -> u32 {
    #[cfg(target_arch = "x86_64")]
    if is_x86_feature_detected!("avx2") {
        // SAFETY: 已检测到 avx2 特性
        return unsafe { sum_avx2(slice) };
    }

    slice.iter().sum()
}

#[target_feature(enable = "avx2")]
fn sum_avx2(slice: &[u32]) -> u32 {
    let (prefix, middle, tail) = unsafe { slice.align_to::<__m256i>() };

    let mut sum = prefix.iter().sum::<u32>() + tail.iter().sum::<u32>();

    let mut base = _mm256_setzero_si256();
    for e in middle {
        base = _mm256_add_epi32(base, *e);
    }

    let base: [u32; 8] = unsafe { std::mem::transmute(base) };
    sum += base.iter().sum::<u32>();

    sum
}

asm! 跳转到 Rust 代码

内联汇编 asm! 现可跳转至 Rust 代码中的标号块,使得在内核或硬件交互场景下的低级优化更灵活:

rust 复制代码
unsafe {
    asm!(
        "jmp {}",
        label {
            println!("Jumped from asm!");
        }
    );
}

详细说明参见 the reference。

在 trait 定义中精确捕获泛型 / 生命周期(+ use<...>) 本版本稳定了在 trait 方法返回类型为 impl Trait 时,显式指定应捕获的泛型与生命周期:

rust 复制代码
trait Foo {
    fn method<'a>(&'a self) -> impl Sized;

    // 反糖后:
    // type Implicit1<'a>: Sized;
    // fn method_desugared<'a>(&'a self) -> Self::Implicit1<'a>;

    fn precise<'a>(&'a self) -> impl Sized + use<Self>;

    // 反糖后:
    // type Implicit2: Sized;
    // fn precise_desugared<'a>(&'a self) -> Self::Implicit2;
}

稳定 API 一览

  • Vec::extract_if
  • vec::ExtractIf
  • LinkedList::extract_if
  • linked_list::ExtractIf
  • <[T]>::split_off
  • <[T]>::split_off_mut
  • <[T]>::split_off_first
  • <[T]>::split_off_first_mut
  • <[T]>::split_off_last
  • <[T]>::split_off_last_mut
  • String::extend_from_within
  • os_str::Display
  • OsString::display
  • OsStr::display
  • io::pipe
  • io::PipeReader
  • io::PipeWriter
  • impl From for OwnedHandle
  • impl From for OwnedHandle
  • impl From for Stdio
  • impl From for Stdio
  • impl From for OwnedFd
  • impl From for OwnedFd
  • Box<MaybeUninit>::write
  • impl TryFrom<Vec> for String
  • <*const T>::offset_from_unsigned
  • <*const T>::byte_offset_from_unsigned
  • <*mut T>::offset_from_unsigned
  • <*mut T>::byte_offset_from_unsigned
  • NonNull::offset_from_unsigned
  • NonNull::byte_offset_from_unsigned
  • ::cast_signed
  • NonZero::::cast_signed。
  • ::cast_unsigned。
  • NonZero::::cast_unsigned。
  • ::is_multiple_of
  • ::unbounded_shl
  • ::unbounded_shr
  • ::unbounded_shl
  • ::unbounded_shr
  • ::midpoint
  • ::from_utf8
  • ::from_utf8_mut
  • ::from_utf8_unchecked
  • ::from_utf8_unchecked_mut 这些以前稳定的 API 现在在 const 上下文中也很稳定:
  • core::str::from_utf8_mut
  • <[T]>::copy_from_slice
  • SocketAddr::set_ip
  • SocketAddr::set_port,
  • SocketAddrV4::set_ip
  • SocketAddrV4::set_port,
  • SocketAddrV6::set_ip
  • SocketAddrV6::set_port
  • SocketAddrV6::set_flowinfo
  • SocketAddrV6::set_scope_id
  • char::is_digit
  • char::is_whitespace
  • <[[T; N]]>::as_flattened
  • <[[T; N]]>::as_flattened_mut
  • String::into_bytes
  • String::as_str
  • String::capacity
  • String::as_bytes
  • String::len
  • String::is_empty
  • String::as_mut_str
  • String::as_mut_vec
  • Vec::as_ptr
  • Vec::as_slice
  • Vec::capacity
  • Vec::len
  • Vec::is_empty
  • Vec::as_mut_slice
  • Vec::as_mut_ptr

移除 i586-pc-windows-msvc 目标

Tier 2 目标 i586-pc-windows-msvc 已被移除。它与更常用的 Tier 1 目标 i686-pc-windows-msvc 的唯一区别是 不要求 SSE2,但 Windows 10(所有 windows 目标的最低系统要求,win7 目标除外)本身就需要 SSE2。

请将原有项目迁移至 i686-pc-windows-msvc。详见 Major Change Proposal。

其它改动 更多 Rust / Cargo / Clippy 变化请查看 everything that changed in Rust, Cargo, and Clippy。

1.87.0 的贡献者 众多贡献者共同缔造了 Rust 1.87.0。没有你们就没有今天的发布,衷心感谢!

相关推荐
我是前端小学生2 天前
一文梳理Rust语言中的可变结构体实例
rust
Source.Liu2 天前
【unitrix数间混合计算】2.21 二进制整数加法计算(bin_add.rs)
rust
Include everything2 天前
Rust学习笔记(二)|变量、函数与控制流
笔记·学习·rust
Source.Liu2 天前
【unitrix数间混合计算】2.20 比较计算(cmp.rs)
rust
许野平2 天前
Rust:构造函数 new() 如何进行错误处理?
开发语言·后端·rust
许野平2 天前
Rust:专业级错误处理工具 thiserror 详解
rust·error·错误处理·result·thiserror
蒋星熠3 天前
Rust 异步生态实战:Tokio 调度、Pin/Unpin 与零拷贝 I/O
人工智能·后端·python·深度学习·rust
Include everything3 天前
Rust学习笔记(一)|Rust初体验 猜数游戏
笔记·学习·rust
华科云商xiao徐3 天前
Rust+Python双核爬虫:高并发采集与智能解析实战
数据库·python·rust