由 naruse 发布于 2023 年 12 月 25 日
我们很高兴地宣布 Ruby 3.3.0 的发布。Ruby 3.3 添加了一个名为 Prism 的新解析器,使用 Lrama 作为解析器生成器,添加了一个名为 RJIT 的新的纯 Ruby JIT 编译器,以及许多性能改进,特别是 YJIT。
Prism
- 引入 Prism 解析器 作为默认 gem
- Prism 是一个可移植、容错且可维护的 Ruby 语言递归下降解析器
- Prism 已准备好投入生产并积极维护,您可以将其替代 Ripper
- 有关于如何使用 Prism 的 详尽的文档
- Prism 既是一个 C 库(将由 CRuby 内部使用),也是一个 Ruby gem(可供任何需要解析 Ruby 代码的工具使用)
- Prism API 中值得注意的方法包括
Prism.parse(source)
返回作为解析结果对象一部分的 ASTPrism.parse_comments(source)
返回注释Prism.parse_success?(source)
如果没有错误则返回 true
- 如果您有兴趣贡献,可以直接在 Prism 仓库 上提交 pull request 或 issue
- 您现在可以使用
ruby --parser=prism
或RUBYOPT="--parser=prism"
来体验 Prism 编译器。请注意,此标志仅用于调试。
使用 Lrama 代替 Bison
- 使用 Lrama LALR 解析器生成器 替换 Bison [特性 #19637]
- 如果您有兴趣,请查看 Ruby 解析器的未来愿景
- Lrama 内部解析器已替换为由 Racc 生成的 LR 解析器,以提高可维护性
- 支持参数化规则
(?, *, +)
,它将在 Ruby parse.y 中使用
YJIT
- 相对于 Ruby 3.2 的重大性能改进
- 改进了对 splat 和 rest 参数的支持。
- 为虚拟机堆栈操作分配了寄存器。
- 编译了更多带有可选参数的调用。还编译了异常处理程序。
- 不支持的调用类型和多态调用站点不再退出到解释器。
- 诸如 Rails
#blank?
之类的基本方法和 专门的#present?
已内联。 Integer#*
、Integer#!=
、String#!=
、String#getbyte
、Kernel#block_given?
、Kernel#is_a?
、Kernel#instance_of?
和Module#===
进行了特别优化。- 编译速度现在比 Ruby 3.2 略快。
- 现在在 Optcarrot 上比解释器快 3 倍以上!
- 与 Ruby 3.2 相比,内存使用量显著改善
- 编译代码的元数据使用更少的内存。
- 当应用程序具有超过 40,000 个 ISEQ 时,
--yjit-call-threshold
会自动从 30 提高到 120。 - 添加了
--yjit-cold-threshold
以跳过编译冷 ISEQ。 - 在 Arm64 上生成更紧凑的代码。
- 代码 GC 现在默认禁用
--yjit-exec-mem-size
被视为硬限制,在该限制处停止编译新代码。- 由于代码 GC 而不会出现性能突然下降。在使用 Pitchfork 重新 fork 的服务器上,具有更好的写时复制行为。
- 如果需要,您仍然可以使用
--yjit-code-gc
启用代码 GC
- 添加
RubyVM::YJIT.enable
,可以在运行时启用 YJIT- 您可以启动 YJIT 而无需修改命令行参数或环境变量。Rails 7.2 将使用此方法 默认启用 YJIT。
- 这也可以用于仅在应用程序完成启动后启用 YJIT。如果您想在启动时禁用 YJIT 的同时使用其他 YJIT 选项,则可以使用
--yjit-disable
。
- 默认情况下提供更多 YJIT 统计信息
- 默认情况下现在提供
yjit_alloc_size
和更多与元数据相关的统计信息。 - 由
--yjit-stats
生成的ratio_in_yjit
统计信息现在在发布版本中可用,不再需要特殊的统计或开发版本即可访问大多数统计信息。
- 默认情况下现在提供
- 添加更多分析功能
- 添加了
--yjit-perf
以方便使用 Linux perf 进行分析。 --yjit-trace-exits
现在支持使用--yjit-trace-exits-sample-rate=N
进行采样
- 添加了
- 更彻底的测试和多个错误修复
RJIT
- 引入了一个纯 Ruby JIT 编译器 RJIT 并替换了 MJIT。
- RJIT 仅支持 Unix 平台上的 x86-64 架构。
- 与 MJIT 不同,它在运行时不需要 C 编译器。
- RJIT 仅用于实验目的。
- 您应该在生产环境中继续使用 YJIT。
- 如果您有兴趣为 Ruby 开发 JIT,请查看 k0kubun 在 RubyKaigi 第三天的演示。
M:N 线程调度器
- 引入了 M:N 线程调度器。[特性 #19842]
- M 个 Ruby 线程由 N 个本机线程(操作系统线程)管理,因此降低了线程创建和管理成本。
- 它可能会破坏 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
个本机线程。
- 由于每个 Ractor 每次只能运行一个 Ruby 线程,因此将使用本机线程数,即
性能改进
defined?(@ivar)
通过对象形状进行了优化。- 现在可以中断诸如
Socket.getaddrinfo
之类的名称解析(在可以使用 pthreads 的环境中)。[特性 #19965] - 垃圾回收器的多项性能改进
- 旧对象引用的年轻对象不再立即提升到老年代。这显著降低了主要 GC 收集的频率。[特性 #19678]
- 引入了一个新的
REMEMBERED_WB_UNPROTECTED_OBJECTS_LIMIT_RATIO
调整变量,用于控制触发主要 GC 收集的不受保护的对象数量。默认设置为0.01
(1%)。这显著降低了主要 GC 收集的频率。[特性 #19571] - 为许多缺少它们的内核类型(特别是
Time
、Enumerator
、MatchData
、Method
、File::Stat
、BigDecimal
和其他几种)实现了写屏障。这显著减少了次要 GC 收集时间和主要 GC 收集频率。 - 大多数核心类现在都使用可变宽度分配,特别是
Hash
、Time
、Thread::Backtrace
、Thread::Backtrace::Location
、File::Stat
、Method
。这使得这些类更快地分配和释放,使用更少的内存并减少堆碎片。 - 已将对弱引用的支持添加到垃圾回收器中。[特性 #19783]
自 3.2 以来的其他值得注意的更改
IRB
IRB 收到了一些增强功能,包括但不限于
- 高级
irb:rdbg
集成,提供与pry-byebug
等效的调试体验(文档)。 - 对
ls
、show_source
和show_cmds
命令的分页器支持。 ls
和show_source
命令提供的更准确和有用的信息。- 使用类型分析的实验性自动完成(文档)。
- 现在可以通过新引入的类 Reline::Face 更改完成对话框中的字体颜色和字体样式(文档)
此外,IRB 还进行了广泛的重构并修复了数十个错误,以便于将来的增强。
有关更详细的更新,请参阅 揭示 Ruby 3.3 的 IRB 的巨大飞跃。
兼容性问题
注意:不包括特性错误修复。
-
在没有普通参数的块中调用没有参数的
it
已弃用。it
将在 Ruby 3.4 中引用第一个块参数。[特性 #18980] -
Regexp::new
现在只接受最多 2 个参数,而不是 3 个。这在 Ruby 3.2 中已弃用。[错误 #18797]
已删除环境变量
删除了以下已弃用的方法。
- 环境变量
RUBY_GC_HEAP_INIT_SLOTS
已被弃用且不再起作用。请改用环境变量RUBY_GC_HEAP_{0,1,2,3,4}_INIT_SLOTS
。[特性 #19785]
标准库兼容性问题
ext/readline
已弃用
- 我们有
reline
,这是一个纯 Ruby 实现,与ext/readline
API 兼容。未来我们将依赖reline
。如果你需要使用ext/readline
,你可以通过 rubygems.org 安装ext/readline
,命令是gem install readline-ext
。 - 我们不再需要安装像
libreadline
或libedit
这样的库。
标准库更新
如果用户在没有将以下 gem 添加到 Gemfile 或 gemspec 的情况下 require
这些 gem,RubyGems 和 Bundler 将发出警告。这是因为它们将在未来的 Ruby 版本中成为捆绑的 gem。
如果你使用 bootsnap gem,此警告将被抑制。我们建议至少运行一次带有 DISABLE_BOOTSNAP=1
环境变量的应用程序。这是此版本的限制。
目标库是
- abbrev
- base64
- bigdecimal
- csv
- drb
- getoptlong
- mutex_m
- nkf
- observer
- racc
- resolv-replace
- rinda
- syslog
添加了以下默认 gem。
- prism 0.19.0
以下默认 gem 已更新。
- RubyGems 3.5.3
- abbrev 0.1.2
- base64 0.2.0
- benchmark 0.3.0
- bigdecimal 3.1.5
- bundler 2.5.3
- cgi 0.4.1
- csv 3.2.8
- date 3.3.4
- delegate 0.3.1
- drb 2.2.0
- english 0.8.0
- erb 4.0.3
- error_highlight 0.6.0
- etc 1.4.3
- fcntl 1.1.0
- fiddle 1.1.2
- fileutils 1.7.2
- find 0.2.0
- getoptlong 0.2.1
- io-console 0.7.1
- io-nonblock 0.3.0
- io-wait 0.3.1
- ipaddr 1.2.6
- irb 1.11.0
- json 2.7.1
- 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.1
- open3 0.2.1
- 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.2
- rdoc 6.6.2
- readline 0.0.4
- reline 0.4.1
- resolv 0.3.0
- rinda 0.2.0
- securerandom 0.3.1
- set 1.1.0
- shellwords 0.2.0
- singleton 0.2.0
- stringio 3.1.0
- strscan 3.0.7
- syntax_suggest 2.0.0
- syslog 0.1.2
- tempfile 0.2.1
- 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
以下捆绑的 gem 已更新。
- minitest 5.20.0
- rake 13.1.0
- test-unit 3.6.1
- rexml 3.2.6
- rss 0.3.0
- net-ftp 0.3.3
- net-imap 0.4.9
- net-smtp 0.4.0
- rbs 3.4.0
- typeprof 0.21.9
- debug 1.9.1
有关默认 gem 或捆绑 gem 的详细信息,请参阅 GitHub 版本,如 Logger 或变更日志。
通过这些更改,自 Ruby 3.2.0 以来,5532 个文件已更改,增加了 326851 行代码,删除了 185793 行代码!
圣诞快乐,节日快乐,享受使用 Ruby 3.3 编程的乐趣!
下载
-
https://cache.ruby-lang.org/pub/ruby/3.3/ruby-3.3.0.tar.gz
SIZE: 22065999 SHA1: 1a7e56851bf29bda1183aca99b3b323c58e0187b SHA256: 96518814d9832bece92a85415a819d4893b307db5921ae1f0f751a9a89a56b7d SHA512: 26074009b501fc793d71a74e419f34a6033c9353433919ca74ba2d24a3de432dbb11fd92c2bc285f0e4d951a6d6c74bf5b69a2ab36200c8c26e871746d6e0fc6
-
https://cache.ruby-lang.org/pub/ruby/3.3/ruby-3.3.0.tar.xz
SIZE: 16345456 SHA1: c8f68e1b0a114b90460a0b44165a3b2f540fa5b6 SHA256: 676b65a36e637e90f982b57b059189b3276b9045034dcd186a7e9078847b975b SHA512: 7959c5753bfa0bfc4d6d74060869aabbe9815c1c97930659da11b917ee0803ddbbd80e869e00c48b8694b4ba48709c3b6493fd045568e36e902616c35ababf01
-
https://cache.ruby-lang.org/pub/ruby/3.3/ruby-3.3.0.zip
SIZE: 26935108 SHA1: a433eef1d7f96daeaf3b4cb842d0ed2dd82e7dc1 SHA256: 0e6563f679dd3694732eb3addf9de681c67b584602ac574376b60e7a509d2cd8 SHA512: a94a85937a14b217c1f4b90d24185289ed4aee79239c4f3eecf8034d3fd34e65ee8d66869473857ed153067188adc9b70c0471e4ebe842c9f98ef60c34090450
什么是 Ruby
Ruby 最初由 Matz(松本行弘)于 1993 年开发,现在作为开源项目开发。它在多个平台上运行,并在世界各地广泛使用,尤其是在 Web 开发中。