`
java-admin
  • 浏览: 1376469 次
  • 性别: Icon_minigender_1
  • 来自: 陕西.西安
社区版块
存档分类
最新评论
阅读更多

http://www.ibm.com/developerworks/cn/opensource/os-cn-rubysbl/

 

 

 

Symbol 是什么

Ruby 是一个强大的面向对象脚本语言(本文所用 Ruby 版本为1.8.6),在 Ruby 中 Symbol 表示“名字”,比如字符串的名字,标识符的名字。

创建一个 Symbol 对象的方法是在名字或者字符串前面加上冒号:


创建 symbol 对象

                
:foo
:test 

:”abc

:”I am a boy”

你可能会问,字符串就是字符串,干吗还有字符串的名字?这是因为在 Ruby 中字符串也是一种对象,即 String 对象。无论其结构还是操作和 Symbol 对象都是不同的。

在 Ruby 中每一个对象都有唯一的对象标识符(Object Identifier),可以通过 object_id 方法来得到一个对象的标识符。我们来看看 Symbol 对象和 String 对象的差别:


Ruby 对象标识符

                
irb(main):001:0> puts :foo.object_id
327458
=> nil
irb(main):002:0> puts :foo.object_id
327458
=> nil
irb(main):003:0> puts :"foo".object_id
327458
=> nil
irb(main):004:0> puts "foo".object_id
24303850
=> nil
irb(main):005:0> puts "foo".object_id
24300010
=> nil
irb(main):006:0> puts "foo".object_id
24296170
=> nil

可以看到,前三行语句中的 :foo (或者 :"foo")都是同一个 Symbol 对象,其 object id 为327458,而后三行中的字符串”foo”都是不同的对象,其 object id 依次为24303850、24300010、24296170。

可见,每个 String 对象都是不同的,即便他们包含了相同的字符串内容;而对于 Symbol 对象,一个名字(字符串内容)唯一确定一个 Symbol 对象。

值得注意的是创建 Symbol 对象的字符串中不能含有’\0’字符,而 String 对象是可以的。


非法 Symbol 字符串

                
irb(main):001:0>  :"fo\0o"
SyntaxError: compile error
(irb):1: symbol cannot contain '\0'
        from (irb):1
irb(main):002:0> :"foo\0"
SyntaxError: compile error
(irb):2: symbol cannot contain '\0'
        from (irb):2
irb(main):003:0> puts "foo\0".object_id
24305140
=> nil
irb(main):004:0> puts "fo\0o".object_id
24301000
=> nil
irb(main):005:0>

除了可以采用一般的字符串,还可以使用操作符(例如+, -, *, /),变量,常量,方法甚至类的名字来创建 Symbol 对象,例如:+就是一个合法的 Symbol 。实际上,在 Ruby 内部操作符、变量等名字本身就是作为 Symbol 处理的,例如当你定义一个实例变量时, Ruby 会自动创建一个 Symbol 对象,例如 @test 对应为 :@test 。


实例变量的 Symbol

                
class Test  			
 attr_accessor :test
end

这个类定义了一个具有读写方法的实例变量 @test 。实际上 Ruby 创建了两个 Symbol ,一个是实例变量的 symbol :@test ,另一个是 :test 。那如果使用字符串对象 ”test” 作为参数呢?也可以,仍然会创建两个 symbol ,:test 和 :@test ,为什么还会创建 :test 呢?这是和Ruby的实现相关的(至少Ruby1.8.6里是这样)。

注意,类变量 @@test 和实例变量 @test 对应的 Symbol 显然是不同的。记住:名字相同,则Symbol 相同


名字相同, Symbol 相同

                
class Test  			
  puts :Test.object_id
  Test = 10
  puts :Test.object_id
  
  def Test  					
    puts :Test.object_id
  end  
end 

Test.new.Test


运行结果
                
224298
224298
224298


名字不同, Symbol 不同
                
class Test  			
  puts :Test.object_id

  @@test = 10
  puts :@@test.object_id
  def test  					
    puts :test.object_id
    @test = 10  				
    puts :@test.object_id

  end  
end 

t =Test.new  
t.test  


运行结果
                
224298
288068
79858
288108

第一个例子里,类名、常量名和方法名都是 Test ,因此相应的 Symbol 对象都是 :Test 。不用担心, Ruby 可以很好区分它在不同上下文中到底表示什么。当然这并不是一个好的编程风格,但对于理解 Ruby 的 Symbol 还是有帮助的: Symbol 表示一个名字,仅此而已。

Symbol 对象一旦定义将一直存在,直到程序执行退出。所有 Symbol 对象存放在 Ruby 内部的符号表中,可以通过类方法 Symbol.all_symbols 得到当前 Ruby 程序中定义的所有 Symbol 对象,该方法返回一个 Symbol 对象数组。由于 Symbol 比较多,你可以 dump 到文件中来查看。


all_symbols 方法

                
irb(main):001:0> Symbol.all_symbols.size
=> 4047
irb(main):002:0> Symbol.all_symbols[0..9]
=> [:@level_notifier, :ppx, :msg_dn, :version, :secs, :@user, :pos, :socketpair,
 :TkENSURE, :HTTPAccepted]
irb(main):003:0> File.open("sym", "w") do |file| file.puts Symbol.all_symbols end
=> nil

Symbol 和 String

Symbol 对象和 String 对象是完全不同的东西,对象标识符很明确的说明了这一点。除此之外,我们还可以从两种对象的方法上区分。

查看 Ruby 库参考,你会发现 String 类有非常多的方法,包括 Mixed-in 方法(Ruby中一个类通过 include 其他模块而得到的方法,实现多重继承的效果)、类方法和实例方法;而 Symbol 类只有一个类方法 all_symbols 和7个实例方法。

例如,可以通过 []= 方法改变 string 的内容,而 symbol 则不行:


[]= 方法比较

                
irb(main):001:0> s="test"
=> "test"
irb(main):002:0> s[0]='1'
=> "1"
irb(main):003:0> puts s
1est
=> nil
irb(main):004:0> sym=:test
=> :test
irb(main):005:0> sym[0]=1
NoMethodError: undefined method `[]=' for :test:Symbol
        from (irb):5
irb(main):006:0>

虽然 Symbol 和 String 是不同的对象,但它们之间关系很密切。 Ruby 提供了方法在 Symbol和 String 之间转换。

Symbol 转化为 String

使用 to_s 或 id2name 方法将 Symbol 转化为一个 String 对象:


Symbol 到 String

                
irb(main):001:0> :test.id2name
=> "test"
irb(main):002:0> :test.to_s
=> "test"
irb(main):003:0> :"I am a boy".to_s
=> "I am a boy"

注意,每个 String 对象都是唯一的,因此对一个 Symbol 调用多次将产生多个 String 对象。

String 转化为 Symbol

除了在字符串前面加冒号,还可以使用 to_sym 或 intern 方法将 String 转化为 Symbol ,如果该 Symbol 已经存在,则直接返回。


String 到 Symbol

                
irb(main):001:0> var1 = "test".to_sym
=> :test
irb(main):002:0> var2 = "test".intern
=> :test
irb(main):003:0> var1 == var2
=> true
irb(main):004:0>

使用 Symbol

正如前边提到的, Ruby 内部一直在使用 Symbol ,比如 Ruby 程序中的各种名字,Symbol本质上是 Ruby 符号表中的东西。使用 Symbol 处理名字可以降低 Ruby 内存消耗,提高执行速度,这点我们在下一篇文章中会看到。

那么 Symbol 对我们有什么用呢?当然也是内存。使用 String 的开销太大了,因为每一个String 都是一个对象。想想前边的例子,一个字符串每出现一次 Ruby 就会创建一个 String 对象。

通常来讲,当你面临 String 还是 Symbol 的选择时,可以参考以下标准:

  • 如果使用字符串的内容,这个内容可能会变化,使用 String
  • 如果使用固定的名字或者说是标识符,使用 Symbol

那么什么时候我们会用到名字呢?很多时候都会,比如枚举值、关键字(哈希表关键字、方法的参数)等等

作为哈希表的 key

哈希表是 Symbol 应用最为广泛的地方。

在ruby中,哈希和数组类似,一个哈希表是一系列 key/value 对的集合,只不过它的 key 取值范围更广泛,可以是任何对象,比如正则表达式。但通常我们都会取有意义的 key ,比如 String、Symbol 。

下面这个哈希表表示按城市分类的一些机器的集合。


一个哈希表例子

                
hosts{
       'beijing' => 'machine1',
       'shanghai'  => 'machine2',
       'guangzhou' => 'machine3',
       'tianjin' =>  'machine4',
       'shenzhen' => 'machine5'
}

如果要引用 beijing 的机器,使用 hosts['beijing'] 。但如果我们程序中要频繁引用哈希表中 value ,这样就不大好了,因为 Ruby 对每一次字符串引用都会生成一个 String 对象,累积下来这个开销是相当大的。

我们完全可以使用 Symbol ,因为对于这些 key 来讲,我们用的就是名字而已,例如下面hosts[:beijing]


使用 Symbol 作为 key
                
hosts = {
 :beijing => 'machine1',
 :shanghai => 'machine2',
 :guangzhou => 'machine3',
 :tianjin  => 'machine4',
 :shenzhen => 'machine5'
}

哈希参数

通常我们定义的函数的参数的个数和顺序是写死的,调用函数的时候要确保参数的个数、顺序匹配,有时候这样很不方便,使用哈希参数可以解决这个问题。

ROR 中就大量地运用这种方式,也许你已经看到了,到处都是 Symbol 和哈希。比如:


使用哈希参数的方法调用

                
link_to 'Show', :action => 'show', :id => product

add_column :products, :price, :decimal, 
:precision => 8, :scale => 2, :default => 0

使用哈希参数的方法可以如下定义,前半部分为固定参数,后面为可变参数,或者干脆全采用哈希参数:


哈希参数

                
def my_method(para1, …, options={})
#your code
end

def my_method(options={})
#your code
end

如果你希望设定一些默认参数,并允许调用者更改这些参数,可以使用哈希对象的 merge! 方法

hsh.merge!( other_hash )该方法将 other_hash 里内容加到 hsh 中,如果other_hash hsh 有重复的 key ,则 key other_hash 中的 value 覆盖 hsh 中对应 key value


方法定义-使用默认参数

                
class Test
 def my_method(opts={})
  default_opts={:arg1 => 10, :arg2 => "abc"}
  default_opts.merge!(opts)
  default_opts.each{|key,value| puts "#{key} is #{value}"}
 end
end

t = Test.new
t.my_method :arg1=>5, :arg3=>"def"


运行结果
                
arg1 is 5
arg2 is abc
arg3 is def


在 Rails 中,对 hash 类进行了扩展,可以使用 reverse_merge! 方法来达到上述效果。该方法在 ActiveSupport::CoreExtensions::Hash::ReverseMerge 中定义。Rails 甚至还提供了 assert_valid_keys 方法,对传递进来的哈希表的 keys 进行合法性检验。

Perl 说,条条大路通罗马。在 Ruby 中也是这样的,也许你会发现更好的应用 Symbol 和哈希的方法。

下篇文章我们将深入 Ruby 内部来看看 Symbol 的样子。


参考资源 (resources)

<!-- CMA ID: 293468 --><!-- Site ID: 10 --><!-- XSLT stylesheet used to transform this file: dw-article-6.0-beta.xsl -->

分享到:
评论

相关推荐

    openGauss Symbol(openGauss-3.0.0-CentOS-64bit-symbol.tar.gz)

    《深入理解openGauss Symbol:探索3.0.0版本的细节》 openGauss,一个开源、高性能的关系型数据库管理系统,旨在为大数据时代提供可靠的数据存储与处理解决方案。openGauss-3.0.0-CentOS-64bit-symbol.tar.gz 文件...

    symbol字体键盘对照表

    symbol字体键盘对照表,直接打出各种希腊字母,不需要公式编辑器,在word里面对照symbol字体直接打出即可

    symbol 刷机USB驱动

    "Symbol"是摩托罗拉解决方案公司的一个品牌,主要生产工业级条形码扫描器、移动数据终端等设备,这些设备在零售、物流、医疗等领域广泛应用。"MC系列"是Symbol推出的一系列坚固型移动计算机,它们配备有各种操作系统...

    字体-Symbol-Tiger-Expert

    "字体-Symbol-Tiger-Expert"这个主题聚焦于一种特定的字体,用于解决在Windows 7系统中因字体文件缺失而导致数学符号无法显示的问题。Symbol字体是一种特殊的符号字体,通常包含了各种数学、化学和科学符号,以及...

    symbol Italic 希腊字母 斜体 希文 斜体

    symbol Italic 希腊字母 斜体 希文 斜体 希腊 斜体 symbol 斜体

    Javascript Symbol原理及使用方法解析

    什么是Symbol JavaScript标准中规定对象的key只能是 String 或 Symbol 类型,区别在于 String 类型的key可以重复而 Symbol 类型的key是唯一的。Symbol 的本质是表示一个唯一标识。每次创建一个Symbol,它所代表的值...

    Symbol条码枪使用手册

    Symbol 条码枪 LS4208 使用手册 Symbol LS4208 是一款高性能的条码枪,广泛应用于零售、物流、医疗、制造等行业。本手册提供了 Symbol LS4208 的使用指南,帮助用户快速上手使用该设备。 1. Symbol LS4208 概述 ...

    SYMBOL扫描枪设置手册

    * Caps Lock 覆盖:键盘将忽略 Caps Lock 按键的状态,因此,条码中的一个大写“A”被传送作为一个大写“A”,无论键盘的 Caps Lock 键的状态是什么。(默认值) 1.2 通过 RS-232 与主机端口连接 * RS-232 主机...

    symbol LS2208

    标题与描述中的关键词“symbol LS2208”指向了由Symbol Technologies公司制造的一款非常流行的条形码扫描器。这款扫描器在零售业、仓库管理、物流和其他需要快速准确读取条形码的行业中被广泛使用。以下是关于symbol...

    openGauss Symbol(openGauss-3.0.0-openEuler-64bit-symbol.tar.gz)

    《深入解析openGauss Symbol:基于openGauss-3.0.0-openEuler-64bit-symbol.tar.gz》 openGauss是一个开源的关系型数据库管理系统,由华为公司主导开发,旨在提供高性能、高可用性、高安全性的企业级数据库解决方案...

    Carrier & Symbol Timing Recovery.zip_carrier中文是什么名字

    在通信系统中,"Carrier" 和 "Symbol Timing Recovery" 是两个至关重要的概念,尤其是在数字通信领域。让我们逐一探讨这两个主题。 首先,"Carrier" 在中文中通常被称为载波。载波是一个固定频率的电信号,它被用来...

    wps_symbol_fonts.zip

    标题 "wps_symbol_fonts.zip" 暗示了这是一个针对WPS Office软件的补丁或资源包,专门解决在Linux操作系统环境下WPS缺少特定字体的问题。这个压缩包包含的字体文件,可能是为了确保WPS在Linux系统中能够正确显示某些...

    Symbol MC1000开发手册

    ### Symbol MC1000 开发手册核心知识点解析 #### 一、版权与许可声明 在文档开头,Symbol Technologies, Inc. 对该手册进行了明确的版权保护声明,并规定了使用条款: - **版权归属**:所有版权归属于Symbol ...

    SYMBOL FACTORY 图符集非常有用

    "SYMBOL FACTORY 图符集" 是一个专为制作自动化方案设计的资源库,其中包含了大量的工业图符,这些图符对于理解和构建自动化系统来说至关重要。这个图符集旨在提高工作效率,帮助工程师们快速、清晰地绘制出各种控制...

    symbol-4278设置手册

    《symbol-4278设置手册》是一份详尽的指南,主要针对Symbol公司的4278系列条形码扫描器的配置与操作。Symbol Technologies是条形码和无线数据通信技术领域的领导者,其产品广泛应用于零售、物流、医疗保健等众多行业...

    Symbol Factory 2.0.138

    "Symbol Factory 2.0.138"是一款专为工业控制(Industrial Control)领域设计的图形库软件,它的主要功能在于提供丰富的图形资源,帮助用户快速创建和编辑各种工控设备、仪器仪表的图标和图形。这款软件的一大特点是...

    工控图库工具symbol-factory-2.0

    《工控图库工具symbol-factory-2.0详解及应用》 在自动化工程和工业控制领域,图形化界面的设计与构建是至关重要的。Symbol Factory 2.0是一款专为工控系统设计的图库工具,它为组态王软件提供强大的辅助功能,帮助...

    SymbolTypeViewer

    "SymbolTypeViewer"是一款专为IT专业人士设计的工具,主要用于查看和分析符号类型。这款软件在调试和排错过程中尤其有用,因为它可以帮助用户理解和解析内存中的符号信息,这对于理解代码执行流程和诊断问题至关重要...

    Symbol 扫描软件

    自己开发的用于在Symbol手持扫描枪上扫描标签 1、安装数据线连接程序:MicrosoftActiveSync_setup_cn(在PC机上安装) 2、向symbol设备安装.net框架类库:NETCFSetupv2(在PC机上安装),安装前应先在PC机上安装好...

    ELF文件symbol分析

    ### ELF文件Symbol分析 在深入探讨ELF(Executable and Linkable Format)文件中符号(symbol)的解析之前,我们首先简要回顾一下ELF文件的基本结构和符号在其中扮演的角色。 #### ELF文件概述 ELF是一种标准的二...

Global site tag (gtag.js) - Google Analytics