您的位置:首页 > 编程语言 > Ruby

Ruby元编程-学习笔记(二)-方法

2016-04-22 23:29 405 查看

动态方法

使用动态方法可以有效的消除重复代码

动态调用方法

class MyClass
def my_method(arg)
arg * 2
end
end

obj = MyClass.new
obj.my_method(3)        # => 6
obj.send(:my_method, 3) # => 6


使用.和send都可以调用my_method方法,不过通过send方法可以将调用的放名作为参数,这样就可以在代码运行期间,直到最后一刻才决定调用哪个方法,这种技术被称为动态派发.

同样由于send方法功能的强大,也带来了一些弊端,他会破坏封装性,示例代码如下

class MyClass
private
def private_method
puts "private_method"
end
end

obj = MyClass.new
obj.private_method          # => 直接调用私有方法会出错
obj.send(:priavet_method)   # => private_method


符号

所有能构成合法变量名的字符串前加上:都为符号, 符号和字符串没有关系,他们属于完全不同的类, 他们的区别在于: 符号不可变, 字符串可变,而且一些操作针对符号运算更加快, 符号用于表示事务的名字,尤其是与元编程相关的名字,比如方法名.另外,通过String#to_sym()或String#intern()方法可以把字符串转换为符号,反之使用Symbol#to_s()或Symbol#id2name.


动态定义方法

# 可使用Module#define_method()方法定义一个方法
class MyClass
def self.define_component(name)
define_method name do
puts "define method #{name}"
end
end
end

obj = MyClass.new
MyClass.define_component :my_method
obj.my_method       # => define method my_method


define_method方法是类方法,只有类才能调用,也是私有方法

1.不能有显式调用,也就是不能有接受者,不能self.define_method这样调用

2.可以通过send(:define_method)强制调用

method_missing方法

class MyClass
def method_missing(method, *args)
puts "You called: #{method}(#{args.join(', ')})"
puts "(You also passed it a block)" if block_given?
end
end

obj = MyClass.new
obj.my_method('a', 'b') {}

=> You called: my_method(a, b)
(You also passed it a block)


method_missing是Kernel的一个实例方法,而所有的对象都继承自Kernel模块, 当调用一个方法时,若在祖先链中未找到该方法,则会调用method_missing方法, Kernel#method_missing()方法会抛出一个NoMethodError进行响应.

幽灵方法

当需要定义很多相似的方法时,可以通过method_missing()方法来方便开发, 使用method_missing方法处理消息,从调用者角度看与普通方法并无差别,但实际上接收者并没有相应的方法,只是统一进行了处理,这被称为幽灵方法.

class MyOpenStruct
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

obj = MyOpenStruct.new
obj.key = "value"
obj.key     # => "value"


上述代码会捕捉obj的调用,并将key或vaule存放在哈希表中.

注:在性能方面,幽灵方法比普通方法稍慢,因为要查找的路径更长.

动态代理

一个捕获幽灵方法调用并把它们转发给另一个对象的对象,称为动态代理.

class D
def data_method(text)
puts "data_method() => #{text}"
end
end

class MyClass
def initialize(data)
@data = data
end
def method_missing(name)
@data.send(:data_method, name)
end
end

obj = MyClass.new(D.new)
obj.aabb        # => data_method() => aabb


注意:使用method_missing技术后,所有不存在方法的调用都会调用你覆盖的方法,这样会导致一些错误信息(你看不到NoSuchMethod这样的提示了),因此, 在使用method_missing方法的时候,一定要限定其使用范围,并且调用父类的super.method_missing方法,还原错误处理.

const_missing方法

与method_missing方法类似, 当引用一个不存在的常量时,Ruby将把这个常量名作为一个符号传递给const_missing方法.

白板类

当一个幽灵方法和真实方法发生冲突时, 而会执行真实的方法, 这个问题就是动态代理技术的通病, 为了安全起见, 应该在代理类中删除绝大多数继承来的方法, 这就是所谓的白板类.

- Module#undef_method()方法会删除所有的(包括继承的)方法

- Module#remove_method()方法只会删除接收者自己的(保留继承的)方法
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  ruby 编程