Ruby 关于 Symbol 对象的文档具有误导性

Ruby documentation over Symbol objects is misleading

我是 Ruby 的新手,和其他人一样,我无法理解 Ruby 符号。我知道这个主题已被多次提及,但我相信这个 post 可能与其他主题略有不同。如果没有,我很抱歉。以 documentation.

中的这段代码为例
module One
  class Fred
  end
  $f1 = :Fred
end
module Two
  Fred = 1
  $f2 = :Fred
end
def Fred()
end
$f3 = :Fred
$f1.object_id   #=> 2514190
$f2.object_id   #=> 2514190
$f3.object_id   #=> 2514190

我的抱怨是它让我们认为在 class、模块或函数和 :Fred 符号之间有一个 link。难怪人们会问 "can I assign a value to a symbol" 或符号是对另一事物的引用。

这段代码增加了混乱:

class TestController < ApplicationController

    layout :which_layout

    def index
        ...
    end

    private

    def which_layout
        if condition
            "layout1"
        else
            "layout2"
        end
    end
end

起初,我以为有一个对该函数的引用,但实际上 layout 方法的行为会根据我们是否传递 String(模板名称)或 Symbol(调用符号指定的方法),如文档所述。 (它是否寻找与我们作为参数传递的符号等效的 method.to_sym?)

但是我相信我已经读到的是,当创建一个 class 时,他的符号对应物将被自动创建,即 :Fred 将已经存在于后续调用中。但仅此而已?

我的问题是:为什么他们必须包含一个 class、一个变量和一个函数来说明这一点?上下文?那为什么要同名呢?为什么不直接做 :

$f1 = :Fred
$f2 = :Fred
$f3 = :Fred

$f1.object_id   #=> 2514190
$f2.object_id   #=> 2514190
$f3.object_id   #=> 2514190

我的理解是,它是为了说明一个事实,即所有其他 "Fred's" 都不会改变符号 :Fred 在任何情况下都不变的事实。也许将示例底部的 object_id 列表更改为类似这样的内容会使它更清楚:

p One::Fred.object_id  #=> 70222371662500
p Two::Fred.object_id  #=> 3
p Fred().object_id     #=> 8
p $f1.object_id        #=> 2514190
p $f2.object_id        #=> 2514190
p $f3.object_id        #=> 2514190

当您使用符号时 Ruby 查看现有符号列表,因此当您重复使用符号时,您不会在内存中创建单独的对象。

irb(main):006:0> :foo.object_id == :foo.object_id
=> true

您可以将其与字符串进行对比:

irb(main):007:0> "foo".object_id == "foo".object_id
=> false

再加上它们比较便宜的事实,使得符号像散列键一样有效。

这个相当令人困惑的例子表明,符号不是范围私有的——符号的 table 是全局的。如果使用实例变量而不是全局变量,那么混淆会少一些。我认为它还试图证明模块和 class 名称是常量。

irb(main):016:0> Fred = Module.new do; end # this is the same as using the module keyword
irb(main):017:0> Fred.object_id != :Fred.object_id
=> true

这意味着 Fred 是对模块的引用,而 :Fred 是一个值(符号)。

像字符串这样的符号是一个值,因此不能用作参考。这与 truefalsenil 非常相似,它们都是单例对象。

irb(main):008:0> true.class
=> TrueClass
irb(main):09:0> true.object_id == true.object_id
=> true
# you can even use the singletons as hash keys
irb(main):010:0> { true => 1, false => 2, nil => 3 }[true]
=> 1

Rails 示例并没有那么复杂。 :which_layout 只是传递给布局方法的参数。布局方法有一个条件,它使用 Object#send 动态调用 :which_layout 方法(如果存在)。而是直接使用字符串参数来构造一个 glob。