当我说{:bla => 1,:bloop => 2}时,:到底做什么?我在什么地方读到过,它与字符串相似,但某种程度上是一个符号。

我对这个概念不是很清楚,有人能给我一些启发吗?


当前回答

这是一个象征。基本上,您说的是散列的两个元素有键bla和bloop,就像您使用了字符串“bla”和“bloop”一样。但是,它们比字符串占用更少的内存,并且更容易输入。

其他回答

所有这些答案都忽略了一个额外的诱人细节。如果你对符号进行字符串化:foo,你会得到..猜猜. .字符串“foo”。因此

irb(main):025:0>
irb(main):026:0> :foo
=> :foo
irb(main):027:0> "#{:foo}"
=> "foo"
irb(main):028:0>
irb(main):029:0> 'foo' <=> :foo
=> nil
irb(main):030:0> 'foo' <=> :foo.to_s
=> 0
irb(main):031:0>

因此. .Perl程序员..这是Ruby对“赤裸的词”的回答。

在ruby中,每个对象都有一个唯一的对象标识符,如果你写入“hello”。Object_id在你的irb和点击返回2次不同的,你将得到2个不同的返回值,但如果你写:hello。Object_id 2次,只能得到相同的返回值。 这应该可以解释这种差异。

如果你使用:foo => bar, foo将是一个符号。符号的好处是它们是独一无二的。当你调用散列中的一个项时,你做hash[:foo]。

符号比字符串需要更少的内存,如果你想让你的程序更快一点,它们也很有用。

这里引用了著名的《敏捷Web开发与Rails》一书中的一些内容,可能对理解这个符号也有帮助:

Rails使用符号来标识事物。特别地,它在命名方法参数和在哈希中查找内容时使用它们作为键。

redirect_to :action => "edit", :id => params[:id]

您可以将符号视为神奇地变成常量的字符串字面量。或者,你可以认为冒号表示“命名的东西”,所以:id是“命名为id的东西”。

只是为了证明答案中提到的一些事情:

require 'benchmark'

n = 1_000_000

print '"foo".equal? "foo" -> ', ("foo".equal? "foo"), "\n"
print '"foo" == "foo"     -> ', ("foo" == "foo"    ), "\n"
print ':foo.equal? :foo   -> ', (:foo.equal? :foo  ), "\n"
print ':foo == :foo       -> ', (:foo == :foo      ), "\n"

Benchmark.bm(10) do |b|
  b.report('string')     { n.times { "foo".equal? "foo" }}
  b.report('str == str') { n.times { "foo" == "foo"     }}
  b.report('symbol')     { n.times { :foo.equal? :foo   }}
  b.report('sym == sym') { n.times { :foo == :foo       }}
end

运行它输出:

"foo".equal? "foo" -> false
"foo" == "foo"     -> true
:foo.equal? :foo   -> true
:foo == :foo       -> true

那么,用equal来比较字符串和字符串?失败是因为它们是不同的对象,即使它们的内容相同。==比较内容,用符号进行等效检查要快得多。

                 user     system      total        real
string       0.370000   0.000000   0.370000 (  0.371700)
str == str   0.330000   0.000000   0.330000 (  0.326368)
symbol       0.170000   0.000000   0.170000 (  0.174641)
sym == sym   0.180000   0.000000   0.180000 (  0.179374)

两种符号测试在速度方面基本相同。在100万次迭代之后,只有0.004733秒的差异,所以我认为这是一次洗涤。