发布 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。没有你们就没有今天的发布,衷心感谢!

相关推荐
独孤留白12 小时前
从C到Rust:Rust 的 Trait 不是Interface,那是什么?
rust
花褪残红青杏小20 小时前
Rust图像处理第7节-马赛克像素化:分块取平均色实现打码风格
rust·webassembly·图形学
doiito1 天前
【Agent Harness】Gliding Horse 设计细节 -- 不跟风开发自己的AI Agent
架构·rust·agent
doiito2 天前
【Agent Harness】Gliding Horse 核心设计理念,不跟风开发自己的AI Agent
ai·rust·架构设计·系统设计·ai agent
花褪残红青杏小2 天前
Rust图像处理第6节- 均值模糊 & 中值模糊:3×3 邻域的两种经典玩法
rust·webassembly·图形学
子兮曰2 天前
前端工具链的「Rust 化」:一场没有赢家的军备竞赛?
前端·后端·rust
星栈2 天前
写 Dioxus Demo 不难,难的是把它写成项目
前端·rust·前端框架
mCell2 天前
【锐评】桌面端技术营销:别拿跑分当工程判断
前端·rust·electron
武子康2 天前
调查研究-201 Rust 里的 dev build 和 release build:为什么同一份代码性能差这么多?
后端·架构·rust
doiito3 天前
【Agent Harness】Gliding Horse 的 L2 作战地图:让多 Agent 协作从“摸黑”变成“透明”
ai·rust·架构设计·系统设计·ai agent