NEWS for Ruby 2.7.0¶ ↑
本文档是版本之间用户可见的功能变更列表,不包括错误修复。
请注意,每个条目都保持简洁,没有提供背后的原因或参考信息。有关包含所有充分信息的更改的完整列表,请参阅 ChangeLog 文件或 Redmine(例如,https://bugs.ruby-lang.org/issues/$FEATURE_OR_BUG_NUMBER
)。
自 2.6.0 版本以来的更改¶ ↑
语言更改¶ ↑
模式匹配¶ ↑
-
模式匹配作为实验性功能引入。[特性 #14912]
case [0, [1, 2, 3]] in [a, [b, *c]] p a #=> 0 p b #=> 1 p c #=> [2, 3] end case {a: 0, b: 1} in {a: 0, x: 1} :unreachable in {a: 0, b: var} p var #=> 1 end case -1 in 0 then :unreachable in 1 then :unreachable end #=> NoMatchingPatternError json = <<END { "name": "Alice", "age": 30, "children": [{ "name": "Bob", "age": 2 }] } END JSON.parse(json, symbolize_names: true) in {name: "Alice", children: [{name: name, age: age}]} p name #=> "Bob" p age #=> 2 JSON.parse(json, symbolize_names: true) in {name: "Alice", children: [{name: "Charlie", age: age}]} #=> NoMatchingPatternError
-
有关更多详细信息,请参阅以下幻灯片
-
可以使用-W:no-experimental option来抑制针对模式匹配的警告。
关键字参数的规范已向 3.0 靠拢¶ ↑
-
关键字参数和位置参数的自动转换已弃用,并且转换将在 Ruby 3 中删除。[特性 #14183]
-
当方法调用在最后一个参数处传递一个
Hash
,并且当它没有传递关键字时,并且当被调用的方法接受关键字时,会发出警告。要继续将哈希视为关键字,请添加双星号运算符以避免警告并确保在 Ruby 3 中的正确行为。def foo(key: 42); end; foo({key: 42}) # warned def foo(**kw); end; foo({key: 42}) # warned def foo(key: 42); end; foo(**{key: 42}) # OK def foo(**kw); end; foo(**{key: 42}) # OK
-
当方法调用将关键字传递给接受关键字的方法,但它没有传递足够的位置参数时,关键字将被视为最终的必需位置参数,并且会发出警告。将参数作为哈希而不是关键字传递,以避免警告并确保在 Ruby 3 中的正确行为。
def foo(h, **kw); end; foo(key: 42) # warned def foo(h, key: 42); end; foo(key: 42) # warned def foo(h, **kw); end; foo({key: 42}) # OK def foo(h, key: 42); end; foo({key: 42}) # OK
-
当方法接受特定的关键字但不接受关键字展开时,并且将包含
Symbol
和非 Symbol 键的哈希或关键字展开传递给该方法时,哈希将继续被拆分,并且会发出警告。您需要更新调用代码以传递单独的哈希,以确保在 Ruby 3 中的正确行为。def foo(h={}, key: 42); end; foo("key" => 43, key: 42) # warned def foo(h={}, key: 42); end; foo({"key" => 43, key: 42}) # warned def foo(h={}, key: 42); end; foo({"key" => 43}, key: 42) # OK
-
如果方法不接受关键字,并且使用关键字调用,则关键字仍然被视为位置哈希,没有警告。此行为将在 Ruby 3 中继续工作。
def foo(opt={}); end; foo( key: 42 ) # OK
-
-
如果方法接受任意关键字,则允许将非符号作为关键字参数键。[特性 #14183]
-
在 2.6.0 中禁止在关键字参数哈希中使用非 Symbol 键,但现在再次允许。[错误 #15658]
def foo(**kw); p kw; end; foo("str" => 1) #=> {"str"=>1}
-
-
**nil
允许在方法定义中显式标记该方法不接受关键字。使用关键字调用此类方法将导致ArgumentError
。[特性 #14183]def foo(h, **nil); end; foo(key: 1) # ArgumentError def foo(h, **nil); end; foo(**{key: 1}) # ArgumentError def foo(h, **nil); end; foo("str" => 1) # ArgumentError def foo(h, **nil); end; foo({key: 1}) # OK def foo(h, **nil); end; foo({"str" => 1}) # OK
-
将空关键字展开传递给不接受关键字的方法不再传递空哈希,除非空哈希是必需参数所必需的,在这种情况下会发出警告。删除双星号以继续传递位置哈希。[特性 #14183]
h = {}; def foo(*a) a end; foo(**h) # [] h = {}; def foo(a) a end; foo(**h) # {} and warning h = {}; def foo(*a) a end; foo(h) # [{}] h = {}; def foo(a) a end; foo(h) # {}
-
以上警告也可以使用 -W:no-deprecated option 来抑制。
编号参数¶ ↑
-
引入了作为默认块参数的编号参数。[特性 #4475]
[1, 2, 10].map { _1.to_s(16) } #=> ["1", "2", "a"] [[1, 2], [3, 4]].map { _1 + _2 } #=> [3, 7]
您仍然可以定义一个名为
_1
等的局部变量,并且当存在时会受到尊重,但会发出警告。_1 = 0 #=> warning: `_1' is reserved for numbered parameter; consider another name [1].each { p _1 } # prints 0 instead of 1
不带块的 proc/lambda 已弃用¶ ↑
-
在调用了块的方法中使用没有块的
Proc.new
和Kernel#proc
现在会显示警告。def foo proc end foo { puts "Hello" } #=> warning: Capturing the given block using Kernel#proc is deprecated; use `&block` instead
可以使用-W:no-deprecated option来抑制此警告。
-
在调用了块的方法中使用没有块的
Kernel#lambda
会引发异常。def bar lambda end bar { puts "Hello" } #=> tried to create Proc object without a block (ArgumentError)
其他杂项更改¶ ↑
-
实验性地引入了无始范围。它可能在
case
、Comparable#clamp
的新调用序列、常量和 DSL 中很有用。[特性 #14799]ary[..3] # identical to ary[0..3] case RUBY_VERSION when ..."2.4" then puts "EOL" # ... end age.clamp(..100) where(sales: ..100)
-
将
$;
设置为非 nil 值现在会显示警告。[特性 #14240] 这包括在String#split
中的用法。可以使用-W:no-deprecated option来抑制此警告。 -
将
$,
设置为非 nil 值现在会显示警告。[特性 #14240] 这包括在Array#join
中的用法。可以使用-W:no-deprecated option来抑制此警告。 -
带引号的 here-document 标识符必须在同一行内结束。
<<"EOS " # This had been warned since 2.4; Now it raises a SyntaxError EOS
-
翻转-翻转语法的弃用已恢复。[特性 #5400]
-
现在可以在流畅的点之间放置注释行。
foo # .bar .baz # => foo.baz
-
现在允许使用字面量
self
作为接收者来调用私有方法。[特性 #11297] [特性 #16123] -
修饰符 rescue 现在对于多重赋值和单次赋值的操作方式相同。[错误 #8279]
a, b = raise rescue [1, 2] # Previously parsed as: (a, b = raise) rescue [1, 2] # Now parsed as: a, b = (raise rescue [1, 2])
-
单例类语法中的
yield
现在会显示警告。此行为很快将被弃用。[特性 #15575]。def foo class << Object.new yield #=> warning: `yield' in class syntax will not be supported from Ruby 3.0. [Feature #15575] end end foo { p :ok }
可以使用-W:no-deprecated option来抑制此警告。
-
引入了通过
(...)
进行参数转发。[特性 #16253]def foo(...) bar(...) end
foo
的所有参数都会转发到bar
,包括关键字和块参数。请注意,括号是强制性的。bar ...
被解析为无尽范围。 -
访问和设置
$SAFE
现在总是会显示警告。$SAFE
将在 Ruby 3.0 中成为普通的全局变量。[特性 #16131] -
C API 中的
Object#{taint,untaint,trust,untrust}
和相关函数不再起作用(所有对象始终被认为未受污染),现在将在详细模式下显示警告。即使在 Ruby 3.0 的非详细模式下也会禁用此警告,并且这些方法和 C 函数将在 Ruby 3.2 中删除。[特性 #16131] -
改进发生在
Object#method
和Module#instance_method
。[特性 #15373]
命令行选项¶ ↑
Warning
选项¶ ↑
-W
选项已使用以下 :
扩展,以管理分类的警告。[特性 #16345] [特性 #16420]
-
要抑制弃用警告
$ ruby -e '$; = ""' -e:1: warning: `$;' is deprecated $ ruby -W:no-deprecated -e '$; = //'
-
它与
RUBYOPT
环境变量一起使用$ RUBYOPT=-W:no-deprecated ruby -e '$; = //'
-
要抑制实验性功能警告
$ ruby -e '0 in a' -e:1: warning: Pattern matching is experimental, and the behavior may change in future versions of Ruby! $ ruby -W:no-experimental -e '0 in a'
-
要使用
RUBYOPT
同时抑制两者,请设置空格分隔的值$ RUBYOPT='-W:no-deprecated -W:no-experimental' ruby -e '($; = "") in a'
核心类更新(仅限突出的部分)¶ ↑
Array
- 新方法
-
添加了
Array#intersection
。[特性 #16155] -
添加了
Array#minmax
,其实现速度比Enumerable#minmax
快。[错误 #15929]
-
Comparable
- 修改的方法
-
Comparable#clamp
现在接受Range
参数。[特性 #14784]-1.clamp(0..2) #=> 0 1.clamp(0..2) #=> 1 3.clamp(0..2) #=> 2 # With beginless and endless ranges: -1.clamp(0..) #=> 0 3.clamp(..2) #=> 2
-
Complex
- 新方法
-
添加了 Complex#<=>。因此
0 <=> 0i
不会引发NoMethodError
。[错误 #15857]
-
Dir
Encoding
- 新编码
-
添加了新的编码 CESU-8。[特性 #15931]
-
Enumerable
- 新方法
-
添加了
Enumerable#filter_map
。[特性 #15323][1, 2, 3].filter_map {|x| x.odd? ? x.to_s : nil } #=> ["1", "3"]
-
添加了
Enumerable#tally
。[特性 #11076]["A", "B", "C", "B", "A"].tally #=> {"A"=>2, "B"=>2, "C"=>1}
-
Enumerator
- 新方法
-
添加了
Enumerator.produce
,用于从任何自定义数据转换生成Enumerator
。[特性 #14781]require "date" dates = Enumerator.produce(Date.today, &:succ) #=> infinite sequence of dates dates.detect(&:tuesday?) #=> next Tuesday
-
添加了
Enumerator::Lazy#eager
,它从惰性枚举器生成非惰性枚举器。[特性 #15901]a = %w(foo bar baz) e = a.lazy.map {|x| x.upcase }.map {|x| x + "!" }.eager p e.class #=> Enumerator p e.map {|x| x + "?" } #=> ["FOO!?", "BAR!?", "BAZ!?"]
-
添加了
Enumerator::Yielder#to_proc
,以便 Yielder 对象可以直接作为块参数传递给另一个方法。[特性 #15618] -
添加了
Enumerator::Lazy#with_index
为惰性的。以前,未定义Enumerator::Lazy#with_index
,因此它从Enumerator
中选取了默认实现,而不是惰性的。[错误 #7877]("a"..).lazy.with_index(1) { |it, index| puts "#{index}:#{it}" }.take(3).force # => 1:a # 2:b # 3:c
-
Fiber
- 新方法
-
添加了
Fiber#raise
,其行为类似于Fiber#resume
,但在恢复的光纤上引发异常。[特性 #10344]
-
File
- 新方法
-
添加了
File.absolute_path?
,以可移植的方式检查路径是否为绝对路径。[特性 #15868]File.absolute_path?("/foo") # => true (on *nix) File.absolute_path?("C:/foo") # => true (on Windows) File.absolute_path?("foo") # => false
-
- 修改的方法
-
File.extname
现在在非 Windows 平台上,对于以点号结尾的文件名会返回一个点号字符串。[Bug #15267]File.extname("foo.") #=> "."
-
FrozenError
- 新方法
-
添加了
FrozenError#receiver
方法,用于返回尝试修改的冻结对象。为了在 Ruby 代码中抛出FrozenError
时设置此对象,FrozenError.new
接受一个:receiver
选项。[Feature #15751]
-
GC
- 新方法
-
添加了 GC.compact 方法用于压缩堆。此函数会压缩堆中的活动对象,以便可以使用更少的页面,并且堆可以对 CoW(写时复制)更加友好。[Feature #15626]
有关算法和注意事项的详细信息,请参见:bugs.ruby-lang.org/issues/15626
-
IO
- 新方法
-
添加了
IO#set_encoding_by_bom
方法,用于检查 BOM 并设置外部编码。[Bug #15210]
-
Integer
- 修改的方法
-
Integer#[]
现在支持范围操作。[Feature #8842]0b01001101[2, 4] #=> 0b0011 0b01001100[2..5] #=> 0b0011 0b01001100[2...6] #=> 0b0011 # ^^^^
-
Method
- 修改的方法
-
Method#inspect
显示更多信息。[Feature #14145]
-
Module
- 新方法
-
添加了
Module#const_source_location
方法,用于检索常量定义的位置。[Feature #10771] -
添加了
Module#ruby2_keywords
方法,用于将方法标记为通过常规参数 splat 传递关键字参数,这在将所有参数委托给另一个方法时很有用,并且可以与旧版本的 Ruby 向后兼容。[Bug #16154]
-
- 修改的方法
-
Module#autoload?
现在接受一个可选参数inherit
,就像Module#const_defined?
一样。[Feature #15777] -
Module#name
现在始终返回一个冻结的String
。对于给定的Module
,返回的String
始终相同。此更改是实验性的。[Feature #16150]
-
NilClass
/TrueClass
/FalseClass
- 修改的方法
-
NilClass#to_s
、TrueClass#to_s
和FalseClass#to_s
现在始终返回一个冻结的String
。对于每个值,返回的String
始终相同。此更改是实验性的。[Feature #16150]
-
ObjectSpace::WeakMap
- 修改的方法
-
ObjectSpace::WeakMap#[]=
现在接受特殊对象作为键或值。[Feature #16035]
-
Proc
- 新方法
-
添加了
Proc#ruby2_keywords
方法,用于将 proc 标记为通过常规参数 splat 传递关键字参数,这在将所有参数委托给另一个方法或 proc 时很有用,并且可以与旧版本的 Ruby 向后兼容。[Feature #16404]
-
Range
- 新方法
-
添加了
Range#minmax
方法,其实现比Enumerable#minmax
更快。它返回的最大值现在与Range#max
相对应。[Bug #15807]
-
- 修改的方法
-
Range#===
现在也对String
参数使用Range#cover?
(在 Ruby 2.6 中,对于除字符串之外的所有类型,它从Range#include?
更改而来)。[Bug #15449]
-
RubyVM
- 删除方法
-
RubyVM.resolve_feature_path
已移至$LOAD_PATH.resolve_feature_path
。[Feature #15903] [Feature #15230]
-
String
- Unicode
-
将 Unicode 版本和 Emoji 版本从 11.0.0 更新到 12.0.0。[Feature #15321]
-
将 Unicode 版本更新到 12.1.0,添加了对 U+32FF SQUARE ERA NAME REIWA 的支持。[Feature #15195]
-
将 Unicode Emoji 版本更新到 12.1。[Feature #16272]
-
Symbol
- 新方法
-
添加了
Symbol#start_with?
和Symbol#end_with?
方法。[Feature #16348]
-
Time
- 新方法
-
添加了
Time#ceil
方法。[Feature #15772] -
添加了
Time#floor
方法。[Feature #15653]
-
- 修改的方法
-
Time#inspect
与Time#to_s
分开,并且它会显示时间的亚秒部分。[Feature #15958]
-
UnboundMethod
- 新方法
-
添加了
UnboundMethod#bind_call
方法。[Feature #15955]umethod.bind_call(obj, ...)
在语义上等同于umethod.bind(obj).call(...)
。此惯用法在某些库中用于调用被覆盖的方法。添加的方法在不分配中间Method
对象的情况下执行相同的操作。class Foo def add_1(x) x + 1 end end class Bar < Foo def add_1(x) # override x + 2 end end obj = Bar.new p obj.add_1(1) #=> 3 p Foo.instance_method(:add_1).bind(obj).call(1) #=> 2 p Foo.instance_method(:add_1).bind_call(obj, 1) #=> 2
-
Warning
- 新方法
-
添加了
Warning.[]
和Warning.[]=
来管理发出/抑制某些类别的警告。[Feature #16345] [Feature #16420]
-
- $LOAD_PATH
- 新方法
-
添加了
$LOAD_PATH.resolve_feature_path
。[Feature #15903] [Feature #15230]
-
标准库更新(仅限未完成的更新)¶ ↑
- Bundler
-
升级到 Bundler 2.1.2。请参阅 github.com/bundler/bundler/releases/tag/v2.1.2
-
- CGI
-
当至少存在一个转义字符时,CGI.escapeHTML 的速度提高 2~5 倍。请参阅 github.com/ruby/ruby/pull/2226
-
- CSV
-
升级到 3.1.2。请参阅 github.com/ruby/csv/blob/master/NEWS.md。
-
- Date
-
Date.jisx0301、Date#jisx0301 和 Date.parse 支持新的日本时代。[Feature #15742]
-
- Delegator
-
Object#DelegateClass 接受一个代码块,并在返回的类的上下文中对其进行 module_eval,类似于
Class.new
和Struct.new
。
-
- ERB
-
禁止序列化 ERB 实例。
-
- IRB
-
引入受 Pry gem 启发的语法高亮,用于 Binding#irb 源代码行、REPL 输入以及一些核心类对象的 inspect 输出。
-
引入由 Reline 提供的多行编辑模式。
-
完成时显示文档。
-
默认启用自动缩进和保存/加载历史记录。
-
- JSON
-
升级到 2.3.0。
-
- Net::FTP
-
添加了 Net::FTP#features 来检查可用的功能,以及 Net::FTP#option 来启用/禁用每个功能。[Feature #15964]
-
- Net::HTTP
-
向 Net::HTTP#start 添加了可选参数
ipaddr
,以替换 TCP/IP 连接的地址。[Feature #5180]
-
- Net::IMAP
-
添加了服务器名称指示 (SNI) 支持。[Feature #15594]
-
- open-uri
-
在
Kernel
中警告 open-uri 的“open”方法。请改用 URI.open。[Misc #15893] -
“text/*”媒体类型的默认字符集是 UTF-8 而不是 ISO-8859-1。[Bug #15933]
-
- OptionParser
-
现在为未知选项显示“您是不是要找?”。[Feature #16256]
test.rb
require "optparse" OptionParser.new do |opts| opts.on("-f", "--foo", "foo") {|v| } opts.on("-b", "--bar", "bar") {|v| } opts.on("-c", "--baz", "baz") {|v| } end.parse!
示例
$ ruby test.rb --baa Traceback (most recent call last): test.rb:7:in `<main>': invalid option: --baa (OptionParser::InvalidOption) Did you mean? baz bar
-
- Pathname
-
Pathname.glob 现在将 3 个参数委托给
Dir.glob
以接受base
关键字。[Feature #14405]
-
- Racc
-
从上游存储库合并 1.4.15 并添加了 racc 的 cli。
-
- Reline
-
新的标准库,与 readline 标准库兼容,但以纯 Ruby 实现。它还提供多行编辑模式。
-
- REXML
-
升级到 3.2.3。请参阅 github.com/ruby/rexml/blob/master/NEWS.md。
-
- RSS
-
升级到 RSS 0.2.8。请参阅 github.com/ruby/rss/blob/master/NEWS.md。
-
- RubyGems
- StringScanner
-
升级到 1.0.3。请参阅 github.com/ruby/strscan/blob/master/NEWS.md。
-
兼容性问题(不包括功能错误修复)¶ ↑
-
以下库不再是捆绑的 gem。请安装相应的 gem 以使用这些功能。
-
CMath (cmath gem)
-
Scanf (scanf gem)
-
Shell (shell gem)
-
Synchronizer (sync gem)
-
ThreadsWait (thwait gem)
-
E2MM (e2mmap gem)
-
Proc
-
Proc#to_s
格式已更改。[Feature #16101]
-
Range
-
Range#minmax
过去会遍历范围以确定最大值。它现在使用与Range#max
相同的算法。在极少数情况下(例如,浮点数或字符串的范围),这可能会产生不同的结果。[Bug #15807]
-
标准库兼容性问题(不包括功能错误修复)¶ ↑
-
将标准库提升为默认 gem
-
以下默认 gem 已发布在 rubygems.org 上
-
benchmark
-
cgi
-
delegate
-
getoptlong
-
net-pop
-
net-smtp
-
open3
-
pstore
-
readline
-
readline-ext
-
singleton
-
-
以下默认 gem 仅在 ruby-core 中得到提升,但尚未发布在 rubygems.org 上。
-
monitor
-
observer
-
timeout
-
tracer
-
uri
-
yaml
-
-
-
did_you_mean
gem 已从捆绑的 gem 提升为默认 gem
- pathname
-
当使用 Pathname 参数调用 Kernel#Pathname 时,现在返回该参数而不是创建新的 Pathname。这与其他
Kernel
方法更相似,但可能会破坏修改返回值并期望不修改参数的代码。
-
- profile.rb, Profiler__
-
已从标准库中删除。自 Ruby 2.0.0 以来,它一直未维护。
-
C API 更新¶ ↑
-
已添加许多
*_kw
函数,用于设置传递的最后一个参数是否应被视为关键字。您可能需要切换到这些函数以避免关键字参数分离警告,并确保在 Ruby 3 中行为正确。 -
rb_scan_args 格式字符串中的
:
字符现在被视为关键字参数。传递位置哈希而不是关键字参数将发出弃用警告。 -
带有
ANYARGS
的 C API 声明已更改为不使用ANYARGS
。请参阅 github.com/ruby/ruby/pull/2404
实现改进¶ ↑
Fiber
-
允许使用
--with-coroutine=
选择不同的协程实现,例如:$ ./configure --with-coroutine=ucontext $ ./configure --with-coroutine=copy
-
用纤程池缓存替换之前的堆栈缓存。纤程池在单个内存区域中分配多个堆栈。堆栈分配变为 O(log N),纤程创建摊销为 O(1)。在微基准测试中,性能提高了约 10 倍。github.com/ruby/ruby/pull/2224
-
File
-
File.realpath
现在在许多平台上使用 realpath(3),这可以显著提高性能。[Feature #15797]
-
Hash
-
更改小型
Hash
对象的数据结构。[Feature #15602]
-
- Monitor
-
Monitor 类是用 C 扩展编写的。[Feature #16255]
-
Thread
-
VM 堆栈内存分配现在与本机线程堆栈相结合,从而提高了线程分配性能并减少了与分配相关的故障。在微基准测试中,性能提高了约 10 倍。
-
- JIT
-
当优化假设失效时,JIT 编译的代码会被重新编译为优化程度较低的代码。
-
当方法被认为是纯方法时,将执行
Method
内联。此优化仍处于实验阶段,并且许多方法尚未被认为是纯方法。 -
--jit-max-cache
的默认值已从 1,000 更改为 100。 -
--jit-min-calls
的默认值已从 5 更改为 10,000。
-
RubyVM
-
自 1.9 左右就存在的每个调用站方法缓存得到了改进:缓存命中率从 89% 提高到 94%。请参阅 github.com/ruby/ruby/pull/2583
-
RubyVM::InstructionSequence
-
RubyVM::InstructionSequence#to_binary
方法生成编译的二进制文件。二进制文件大小已减小。[Feature #16163]
-
其他更改¶ ↑
-
已移除对 IA64 架构的支持。 难以找到用于测试的硬件,本地光纤代码难以实现,并且它给解释器增加了不小的复杂性。[功能 #15894]
-
要求编译器支持 C99。[杂项 #15347]
-
我们所用方言的详细信息:bugs.ruby-lang.org/projects/ruby-master/wiki/C99
-
-
Ruby 的上游仓库已从 Subversion 更改为 Git。
-
RUBY_DESCRIPTION 包括 Git 版本,而不是 Subversion 的版本。
-
支持使用
__builtin_
语法在 Ruby 中构建内置方法。[功能 #16254]一些方法在 *.rb 文件中定义(例如 trace_point.rb)。例如,可以轻松定义一个接受关键字参数的方法。