Ruby 3.3.0-rc1 发布

我们很高兴地宣布 Ruby 3.3.0-rc1 的发布。Ruby 3.3 添加了一个名为 Prism 的新解析器,使用 Lrama 作为解析器生成器,添加了一个名为 RJIT 的新的纯 Ruby JIT 编译器,以及许多性能改进,特别是 YJIT。

在 RC1 发布后,我们将尽可能避免引入 ABI 不兼容性。如果需要这样做,我们将在发布说明中声明。

Prism

  • 引入 Prism 解析器 作为默认 gem
    • Prism 是一个可移植的、容错的、可维护的 Ruby 语言递归下降解析器
  • Prism 已准备好用于生产环境并积极维护,您可以将其代替 Ripper 使用
    • 关于如何使用 Prism 的 详细文档
    • Prism 既是一个 C 库,将由 CRuby 内部使用,也是一个 Ruby gem,可供任何需要解析 Ruby 代码的工具使用
    • Prism API 中的重要方法有
      • Prism.parse(source) 它返回 AST 作为 ParseResult 的一部分
      • Prism.dump(source) 它返回序列化的 AST 作为字符串
      • Prism.parse_comments(source) 它返回注释
  • 如果您有兴趣贡献,可以直接在 Prism 存储库上提交 pull 请求或问题

使用 Lrama 代替 Bison

RJIT

  • 引入了纯 Ruby JIT 编译器 RJIT 并替换了 MJIT。
    • RJIT 仅在 Unix 平台上支持 x86-64 架构。
    • 与 MJIT 不同,它在运行时不需要 C 编译器。
  • RJIT 仅用于实验目的。
    • 您应该在生产环境中使用 YJIT。
  • 如果您有兴趣为 Ruby 开发 JIT,请查看 k0kubun 在 RubyKaigi 第三天的演讲

YJIT

  • 与 3.2 相比,性能有了重大改进
    • 改进了对 splat 和 rest 参数的支持。
    • 为虚拟机的堆栈操作分配了寄存器。
    • 编译了更多带有可选参数的调用。
    • 还编译了异常处理程序。
    • 具有巨型对象形状的实例变量不再退出到解释器。
    • 不支持的调用类型不再退出到解释器。
    • Integer#!=String#!=Kernel#block_given?Kernel#is_a?Kernel#instance_of?Module#=== 经过了特别优化。
    • 现在在 optcarrot 上比解释器快 3 倍以上!
  • 与 3.2 相比,内存使用率显著提高
    • 已编译代码的元数据使用更少的内存。
    • 在 ARM64 上生成更紧凑的代码
  • 编译速度现在比 3.2 略快。
  • 添加 RubyVM::YJIT.enable,可以在运行时启用 YJIT
    • 您可以在不修改命令行参数或环境变量的情况下启动 YJIT。
    • 这也可用于仅在应用程序完成启动后启用 YJIT。--yjit-disable 可用于如果您想在启动时禁用 YJIT 的同时使用其他 YJIT 选项。
  • 代码 GC 现在默认禁用,并且 --yjit-exec-mem-size 被视为硬限制,当达到此限制时将停止编译新代码。
    • 在使用 unicorn 和 fork 的服务器上具有更好的写时复制行为
    • 由于代码 GC,性能不会突然下降。
    • 如果需要,您仍然可以使用 --yjit-code-gc 启用代码 GC
  • --yjit-stats 生成的 ratio_in_yjit 统计信息现在在发布版本中可用,不再需要特殊的统计或开发版本才能访问大多数统计信息。
  • 退出跟踪选项现在支持采样
    • --trace-exits-sample-rate=N
  • 添加了 --yjit-perf 以方便使用 Linux perf 进行分析。
  • 更彻底的测试和多个 bug 修复

M:N 线程调度器

  • 引入了 M:N 线程调度器。[Feature #19842]
    • M 个 Ruby 线程由 N 个原生线程(OS 线程)管理,因此减少了线程创建和管理成本。
    • 它可能会破坏 C 扩展兼容性,因此默认情况下在主 Ractor 上禁用 M:N 线程调度器。
      • RUBY_MN_THREADS=1 环境变量在主 Ractor 上启用 M:N 线程。
      • M:N 线程在非主 Ractor 上启用。
    • RUBY_MAX_CPU=n 环境变量设置 N 的最大数量(原生线程的最大数量)。默认值为 8。
      • 由于每个 Ractor 一次只能运行一个 Ruby 线程,因此将使用原生线程的数量,该数量是 RUBY_MAX_CPU 中指定的数量和正在运行的 Ractor 数量中较小的一个。因此,单个 Ractor 应用程序(大多数应用程序)将使用 1 个原生线程。
      • 为了支持阻塞操作,可以使用多于 N 个原生线程。

其他值得注意的新功能

语言

性能改进

  • defined?(@ivar) 通过对象形状进行了优化。
  • 诸如 Socket.getaddrinfo 之类的名称解析现在可以中断(在可以使用 pthreads 的环境中)。Feature #19965
    • 为此,每次调用 getaddrinfo 或 getnameinfo 时都会创建一个 pthread。这会在名称解析中产生一些开销(在我们的实验中约为 2.5 倍)。我们不认为名称解析开销对于大多数应用程序来说是一个问题,但是如果您观察到这种情况,或者如果您看到您认为是由此更改引起的意外影响,请报告它们。
  • 垃圾收集器的多项性能改进
    • 旧对象引用的年轻对象不再立即提升到旧世代。这大大减少了主要 GC 收集的频率。[Feature #19678]
    • 引入了一个新的 REMEMBERED_WB_UNPROTECTED_OBJECTS_LIMIT_RATIO 调整变量,用于控制导致触发主要 GC 收集的未受保护对象数量。默认设置为 0.01 (1%)。这大大减少了主要 GC 收集的频率。[Feature #19571]
    • 为许多缺少它们的内核类型实现了写屏障,特别是 TimeEnumeratorMatchDataMethodFile::StatBigDecimal 和其他一些。这大大减少了次要 GC 收集时间和主要 GC 收集频率。
    • 大多数核心类现在使用可变宽度分配,特别是 HashTimeThread::BacktraceThread::Backtrace::LocationFile::StatMethod。这使得这些类分配和释放更快,使用更少的内存并减少堆碎片。
    • 已将对弱引用的支持添加到垃圾收集器中。[Feature #19783]

自 3.2 以来的其他值得注意的更改

IRB

IRB 收到了多项增强功能,包括但不限于

  • 高级 irb:rdbg 集成,可提供与 pry-byebug 等效的调试体验(文档)。
  • lsshow_sourceshow_cmds 命令的分页器支持。
  • lsshow_source 命令提供的更准确和有用的信息。
  • 使用类型分析的实验性自动完成(文档)。
  • 现在可以通过新引入的类 Reline::Face 更改完成对话框中的字体颜色和字体样式(文档

此外,IRB 还进行了广泛的重构并收到了数十个 bug 修复,以方便未来更容易进行增强。

兼容性问题

注意:不包括功能错误修复。

  • 在没有普通参数的块中,没有参数的 it 调用已被弃用。it 将在 Ruby 3.4 中引用第一个块参数。Feature #18980

已删除的常量

已删除以下已弃用的常量。

已删除的方法

已删除以下已弃用的方法。

已删除的环境变量

已删除以下已弃用的环境变量。

  • 环境变量 RUBY_GC_HEAP_INIT_SLOTS 已被弃用,并且不执行任何操作。请改用环境变量 RUBY_GC_HEAP_{0,1,2,3,4}_INIT_SLOTSFeature #19785

Stdlib 兼容性问题

ext/readline 已停用

  • 我们有 reline,它是一个纯 Ruby 实现,兼容 ext/readline API。 未来我们将依赖 reline。 如果您需要使用 ext/readline,您可以通过 rubygems.org 使用 gem install readline-ext 安装 ext/readline
  • 我们不再需要安装像 libreadlinelibedit 这样的库。

C API 更新

已更新的 C API

以下 API 已更新。

已移除的 C API

以下已弃用的 API 已移除。

标准库更新

如果用户需要未来 Ruby 版本中计划捆绑的 gem,RubyGems 和 Bundler 会发出警告。

目标库是

  • abbrev
  • base64
  • bigdecimal
  • csv
  • drb
  • getoptlong
  • mutex_m
  • nkf
  • observer
  • racc
  • resolv-replace
  • rinda
  • syslog

添加了以下默认 gem。

  • prism 0.15.1

以下默认 gems 已更新。

  • RubyGems 3.5.0.dev
  • base64 0.2.0
  • benchmark 0.3.0
  • bigdecimal 3.1.5
  • bundler 2.5.0.dev
  • cgi 0.4.0
  • csv 3.2.8
  • date 3.3.4
  • delegate 0.3.1
  • drb 2.2.0
  • english 0.8.0
  • erb 4.0.3
  • etc 1.4.3.dev.1
  • fcntl 1.1.0
  • fiddle 1.1.2
  • fileutils 1.7.2
  • find 0.2.0
  • getoptlong 0.2.1
  • io-console 0.6.1.dev
  • irb 1.8.3
  • logger 1.6.0
  • mutex_m 0.2.0
  • net-http 0.4.0
  • net-protocol 0.2.2
  • nkf 0.1.3
  • observer 0.1.2
  • open-uri 0.4.0
  • open3 0.2.0
  • openssl 3.2.0
  • optparse 0.4.0
  • ostruct 0.6.0
  • pathname 0.3.0
  • pp 0.5.0
  • prettyprint 0.2.0
  • pstore 0.1.3
  • psych 5.1.1.1
  • rdoc 6.6.0
  • reline 0.3.9
  • rinda 0.2.0
  • securerandom 0.3.0
  • shellwords 0.2.0
  • singleton 0.2.0
  • stringio 3.0.9
  • strscan 3.0.7
  • syntax_suggest 1.1.0
  • tempfile 0.2.0
  • time 0.3.0
  • timeout 0.4.1
  • tmpdir 0.2.0
  • tsort 0.2.0
  • un 0.3.0
  • uri 0.13.0
  • weakref 0.1.3
  • win32ole 1.8.10
  • yaml 0.3.0
  • zlib 3.1.0

以下捆绑 gem 从默认 gem 提升而来。

  • racc 1.7.3

以下捆绑 gems 已更新。

  • minitest 5.20.0
  • rake 13.1.0
  • test-unit 3.6.1
  • rexml 3.2.6
  • rss 0.3.0
  • net-imap 0.4.4
  • net-smtp 0.4.0
  • rbs 3.2.2
  • typeprof 0.21.8
  • debug 1.8.0

有关默认 gem 或捆绑 gem 的详细信息,请参阅 GitHub 发布版本,例如 Logger 或更新日志。

有关更多详细信息,请参阅 NEWScommit logs

自 Ruby 3.2.0 以来,这些更改导致 5414 个文件发生更改,添加 306141 行代码 (+),删除 183575 行代码 (-)

下载

  • https://cache.ruby-lang.org/pub/ruby/3.3/ruby-3.3.0-rc1.tar.gz

    SIZE: 21783575
    SHA1: c75a860e06f27b7f69b874757417277c21d1d3f4
    SHA256: c4ff82395a90ef76c7f906b7687026e0ab96b094dcf3a532d9ab97784a073222
    SHA512: 265fb2ffe55af47d8349edffcebe749cc170291227cef55529fe4e67363e4e84b88daa34ffb5364a99c8a0e32110266a34c9a11d62f3bd6f6d47fa76eca641f4
    
  • https://cache.ruby-lang.org/pub/ruby/3.3/ruby-3.3.0-rc1.tar.xz

    SIZE: 16163884
    SHA1: 26503f9bdc7d0a05aaa9836f54d3aa9e74a9ead9
    SHA256: 051815637f1fa75a1edf2c54b66d5d5b69563daad777da8dc39543b7754b5290
    SHA512: 82f4acfaad1dc47db27ee3eb952481a95b3474a98059e9e9f5ceb035b690d1faabe99f2ec52371c4089ed1615eb10c395f029088418fec4d26399b65b4f259b9
    
  • https://cache.ruby-lang.org/pub/ruby/3.3/ruby-3.3.0-rc1.zip

    SIZE: 26735443
    SHA1: 35583453a7734216b08829ef0ec9ea1bc0d4ae7f
    SHA256: 56dd82e1dd714f2889ca975ae7befbe005675de08839c2cc4a484de2ae65201c
    SHA512: 7fbe414c230aedc9f364512bcbc213038f8f6e4268929a559d2527e2f3e32a140b394e37098ab7e59161236eca8b89cc9e52d73a3be8d7bd44faa91681483f5d
    

什么是 Ruby

Ruby 最初由 Matz(Yukihiro Matsumoto)于 1993 年开发,现在作为开源项目进行开发。 它可以在多个平台上运行,并在世界各地被广泛使用,尤其是在 Web 开发领域。