`

Ruby之方法的妙用

 
阅读更多
方法的查找与执行
Ruby中对象的方法都定义在类中,当对象想要执行一个方法时,首先需要找到该方法,而Ruby编译器查找方法的方式就是,第一步在自己的类中找,没有的话,就沿着该类的祖先链(ancestors)一直往上找。
String.ancestors        # => [String, Comparable, Object, Kernel, BasicObject]

为什么这儿会出现Comparable和Kernal, 这是因为Module的机制,当一个类include一个模块时,编译会把该模块放在最靠近该类的祖先链上, String类include了Comparable模块,而Kernal则是被Object类include的。

方法执行的时候需要一个接收者,方法就会在接收者对象中被执行,该接收者就是所谓的self对象。一般情况下,self对象是由最后一个接收该方法的对象担当,在类和模块的定义中(并且在任何方法的定义外),self对象由类或模块担任。

动态调用方法
通常方法的调用方式是“对象名.方法名”,在Ruby中有一个很酷的特性,可以通过send()方法,把想调用的方法名作为参数,这样就可以在代码运行时,直到最后时刻才决定调用哪个方法,这种技术称之为动态派发(Dynamic Dispatch)。这个技术非常有用,比方说,当在项目中有一个配置文件对象,会根据配置文件初始化,在使用过程中,不同用户可能会设置不同的值。通常做法是,判断属性的键值是对应到哪个属性,然后,调用对应的set方法,代码如下:
config.name = v if isNameProperty?(k) 
config.password = v if isPasswordProperty?(k) 
config.port = v if isPortProperty?(k) 
... 

看着这么一堆的长得像亲兄弟似的代码,不由得产生一种阉掉它们的冲动。如果使用动态调用方法的话,代码可以简化如下:
load_config('config.properties').each do |k, v| 
config.send("#{k}=", v) 
end 

根据获取的每个键值,去调对应属性的set方法,代码清爽很多,而且以后扩展config对象不需要修改load方法。

动态定义方法
除了动态调用方法外,Ruby甚至支持动态定义方法,通过使用Module#define_method()方法,提供一个方法名和一个充当方法体的块即可定义一个方法。例:
class MyClass 
define_method :doubleString do |args| 
args * 2 
end 
end 
t = MyClass.new 
puts t.doubleString("he")    # => hehe

有了这个黑魔法之后,以后,就可以多个相似方法中不同的部分抽出来作为参数,然后,使用一个方法定义搞定所有的方法。

method_missing()方法
Ruby是动态语言,编译器并不会检测方法调用时的行为,因此你可以调用一个不存在的方法。 在运行时,所有找不到对应方法的调用都会调用一个method_missing()方法,该方法定义在Kernal模块中,因此每个对象都继承了该方法。在kernal中method_missing()方法,会抛出一个NoMethodError的异常,这就是编译器所做的工作。

但是,我们可以覆写这个方法,让它变得很有意思。比方说,创建一个Struct,当你想要新的属性时,只需要给它赋个值就神奇的产生了。
class MyStruct 
def initialize 
@attributes = {} 
end 

def method_missing(name, *args) 
attribute = name.to_s 
if attribute =~ /=$/ 
@attributes[attribute.chop] = args[0] 
else 
@attributes[attribute] 
end 
end 
end 
s = MyStruct.new 
s.weibo = "@xianlinbox" 
puts s.weibo             # => @xianlinbox

这种,从调用者角度看,跟普通方法没什么区别,但是实际接收者却并没有相对应的方法的方法,Ruby术语称之为幽灵方法(Ghost Method)。对于幽灵方法,除了自定义该方法以外,还可以把该方法转发给另外一个对象的方法,当然,你可以在转发前后包装一些自己的逻辑,这种处理技术称之为动态代理(Dynamic Proxy),和前面提到的动态调用异曲同工。

使用了method_missing处理幽灵方法后,所有不存在方法的调用都会到这里来,这可能会导致一些错误信息不直观的问题(你看不到NoSuchMethod这样的提示了),因此, 在使用method_missing方案的时候,一定要限定其使用范围,并且调用父类的super.method_missing方法,让不属于这个范围的,该报什么错还是报什么错。

白板类与保留方法
每一个类都有从父类继承下来的一堆方法,在使用动态代理技术时,如果一个幽灵方法和真实的方法(你没有意料到的继承来的方法)发生名字冲突时,后者会获胜,从而导致系统报错。因此在使用动态代理技术时, 可以在代理类中删除绝大多数继承来的方法,避免发生名字冲突,删除了继承方法的类,就是所谓的白板类(Blank Slate),删除类方法的途径有2个,一个是调用Module#undef_method方法,一个是调用Module#remove_method方法。

在Ruby的Object类中,有一些方法是内部使用的,如果对其重新定义,或删除,可能导致Ruby莫名其妙的挂掉,为了防止这种事情的发生,Ruby在这些方法名前面用“__”打头,这些方法称之为保留方法,当你试图修改这些方法时,Ruby会给出警告。
1
0
分享到:
评论

相关推荐

    Ruby 方法、类

    Ruby还支持块(Blocks),这是一种特殊的匿名方法,通常用`do..end`或花括号`{..}`包裹。块可以与方法结合使用,如`each`迭代器方法: ```ruby [1, 2, 3].each do |num| puts num end ``` ### 类(Classes) 类是...

    Ruby.Programming_向Ruby之父学程序设计(第2版)

    您若是自己动手玩过HTML这样的读者,应该就能充分活用《Ruby Programming:向Ruby之父学程序设计(第2版)》的知识了。另外,若您不是初学者,但想要从头开始学习Ruby语言,这《Ruby Programming:向Ruby之父学程序设计...

    Ruby-Functo是ruby中的可组合方法对象

    标题提到的“Ruby-Functo”就是Ruby中实现可组合方法对象的一个库,它为开发者提供了更灵活和模块化的代码组织方式。 Functo库的设计理念是让函数或方法像对象一样可以被操作、组合和复用。在传统的面向对象编程中...

    Ruby_向ruby之父学程序设计(第二版)(经典入门)

    完整书名《Ruby Programming-向Ruby之父学程序设计》 ,就是那本经典的绿皮书 第一部分:通过简单的Ruby程序来介绍程序的基本架构。  第二部分:介绍基础语法规则,以及类、模块等面向对象程序设计的思考方法与...

    Ruby Ruby Ruby Ruby Ruby Ruby

    Ruby Ruby Ruby Ruby Ruby Ruby

    《Ruby Programming—向Ruby之父学程序设计(第2版)》电子书

    《Ruby Programming—向Ruby之父学程序设计(第2版)》这本书旨在帮助初学者,甚至是完全没有编程背景的读者,快速掌握Ruby语言的基础和高级概念。 首先,Ruby是一种动态类型语言,这意味着变量的类型在运行时自动...

    Ruby-TensorStream用Ruby重新实现TensorFlow

    **Ruby-TensorStream:用Ruby重现实现TensorFlow** Ruby-TensorStream是一个开源项目,旨在为Ruby开发者提供一个类似于Google TensorFlow的深度学习框架。它的核心目标是让Ruby程序员能够利用TensorFlow的强大功能...

    Ruby.Programming_向Ruby之父学程序设计(第2版).pdf (含书签)

    [Ruby.Programming_向Ruby之父学程序设计(第2版)].(日)高桥征义,(日)后藤裕藏.扫描版(ED2000.COM).pdf ) 带书签

    ruby2ruby.zip

    ruby2ruby 提供一些用来根据 RubyParser 兼容的 Sexps 轻松生成纯 Ruby 代码的方法。可在 Ruby 中轻松实现动态语言处理。 标签:ruby2ruby

    06Ruby 方法.docx

    ### Ruby 方法详解 #### Ruby 语言简介 Ruby 是一种动态、面向对象的脚本语言,以其简洁、清晰的语法著称。它支持多种编程范式,包括面向对象、命令式和函数式编程,并且拥有强大的元编程能力。Ruby 的设计哲学强调...

    笨方法学习 Ruby 电子书

    很好的ruby学习材料,ruby的入门有点难,比较繁琐,这本书很好的帮助大家开启ruby的大门

    笨方法学ruby.

    ruby\笨方法学\笨方法学

    笨方法学ruby-中文版

    2. **面向对象编程**:Ruby的核心特性之一就是其强大的面向对象支持。书中会讲解类、对象、继承、模块(mixins)以及方法的动态性,这些都是理解Ruby编程的关键。 3. **异常处理**:Ruby中的异常处理机制允许程序在...

    ruby DBI ruby DBI ruby DBI

    ruby DBI ruby DBI ruby DBIruby DBI ruby DBI ruby DBIruby DBI ruby DBI ruby DBIruby DBI ruby DBI ruby DBIruby DBI ruby DBI ruby DBIruby DBI ruby DBI ruby DBIruby DBI ruby DBI ruby DBIruby DBI ruby DBI ...

    Ruby编程Ruby Programming

    - **动态性**:Ruby支持运行时修改代码结构,允许开发者在程序运行过程中改变类和方法的行为。 - **元编程能力**:Ruby提供了一种称为元编程的技术,使得程序可以自我修改和自我扩展。 ### 二、Ruby编程基础 #### ...

    Ruby-rubyinstall安装RubyJRubyRubiniusMagLevorMRuby

    3. Rubinius:Rubinius是一个用Ruby语言大部分实现的虚拟机,其目标是提供一个高性能的、符合Ruby语言规范的平台。它使用LLVM作为后端,支持即时编译,旨在提供比MRI更好的性能。 4. MagLev:MagLev是GemStone/S...

    Ruby-rubybuild编译和安装Ruby

    Ruby是一种动态、开源的编程语言,以其简洁、优雅的语法和强大的元编程能力著称。在Ruby开发中,为了管理不同版本的Ruby环境,我们常常会使用到`rbenv`和`ruby-build`这两个工具。本文将详细介绍如何使用`ruby-build...

    《Ruby Programming_向Ruby之父学程序设计_第2版》PDF版本下载.txt

    《Ruby Programming_向Ruby之父学程序设计_第2版》PDF版本下载

    Ruby资源ruby-v3.1.1.zip

    6. **弃用警告**:Ruby 3.1对一些即将废弃的功能发出警告,为向未来版本过渡做好准备,比如`Array#bsearch`的旧实现。 7. **兼容性增强**:Ruby 3.1对其他语言的兼容性进行了增强,比如改进了与C++的互操作性。 8....

Global site tag (gtag.js) - Google Analytics