12 种方式轻松实现 Ruby 调用

作者 | Gregory Witek

译者 | 弯月,责编 | 王晓曼

头图 | CSDN 下载自东方IC

出品 | CSDN(ID:CSDNnews)

以下为译文:

最近,与同事聊天的时候,我们谈到了有关 Python 编程的某些方面。我们开玩笑说 Python 之所以能够坚持这种思想,正是因为在 Python 中做每件事都只有一种正确的方法(针对 Python 语言而言,Python 库可不一定)。这不禁让我想到了 Ruby,其编程思想恰恰相反,一切都可以通过许多不同的方式完成。

因此,今天我就来整理一下,在 Ruby 中调用某个方法究竟有多少种方式。最终我找到了12种不同的方式(有一些方式略微有点牵强)。下面我们就来逐一介绍,请做好准备不要太过于吃惊哦,尤其是最后一个肯定会震撼到你!

注意:本文中的代码不适合在生产中使用(尤其是最后3个示例)。这只是我对 Ruby 语言功能的探索。虽然有些技巧在很多情况下都可以派上用场,但请务必谨慎使用。简单性、安全性和可读性远比花哨更为重要。

准备工作

 

为了进行此次实验,我准备了一个类,其中包含了一个方法,下面我将通过多种不同的方式来调用这个方法。为了简单起见,该方法不接受任何参数(不过即使加上参数,每个示例也可以正常工作)。

这个类叫做 User,有一个属性 name,等待被调用的方法名叫 hello,调用这个方法将显示一条欢迎信息,其中包含用户名。

class User
  def initialize(name)
    @name = name
  end

  def hello
    puts "Hello,#{@name}!"
  end

  def method_missing(_)
    hello
  end
end

user = User.new('Gregory')

12种方法

1、最常用的方法 

user.hello()

关于这个方法没什么好说的,相信大量编程语言调用方法时都采用了这种方式。有意思的是,即使在点前后加上空格:user  . hello(),这个调用也依然有效。

2、省略括号

user.hello

严格来说,这种方式与前一种相同,只不过省略了括号,在 Ruby 中这个括号是可选的(只要代码没有歧义不写也没问题;但是当代码可以用多种方式解释时,就必须加上括号)。

3-4、使用 send和 public_send

user.send(:hello)

user.public_send(:hello)

在这两种方式中,我将调用的方法名作为参数传递给 send 和 public_send(每个类都定义了send 和 public_send)。send 和 public_send 之间的区别在于,后者面向的是私有方法。如果在调用私有方法的时候报错,那么依然可以通过 send 调用。

在传递方法名的时候我使用了符号类型:(:hello),但是你也可以使用字符串:("hello")。

5-7、使用 “method” 和 “call”

user.method(:hello).call

user.method(:hello).()

user.method(:hello)[]

在这三个例子中,后两个只是语法糖,所以我把它们放在了一起。这种方式非常有意思。调用 user.method(:hello).call 会返回 Method 类的实例。这个对象可以作为值随意传递,而且也可以随时调用,它还存储了其所属对象的引用,因此,如果修改用户名,那么调用时就会使用新的用户名:

method = user.method(:hello)
user.set_instance_variable(:@name, "Not Only Code")
method.call() # prints "Hello, Not Only Code!"

这里的 .()和 [] 等价于.call(),而且还可以接受参数。proc.call(1,2,3)、 proc.(1,2,3)和 proc[1,2,3]的效果完全相同(尽管最后一个不支持命名参数)。

8、使用 “tap”

user.tap(&:hello)

tap 是一个非常有趣的小方法,它会接受一个块,然后将自身作为参数传递进去并执行该块,最终返回自身。我很少使用它,但是在某些情况下还是很有用的(例如链接方法时的副作用)。

语法 &:hello会将 :hello符号转换为 Proc实例。更多信息请参阅(https://www.honeybadger.io/blog/how-ruby-ampersand-colon-works/)。Proc是一个可调用对象,就像前面示例中的 Method一样。

9、在函数名上使用"to_proc"

:hello.to_proc.call(user)

我喜欢这种方式,因为这种调用反转了顺序:user 变成了函数的参数。实际上这种方式与上一个非常相似:Proc 的 call 函数将初始符号传递给接收到的参数。类似于如下代码:

class Proc
  def call(obj)
   obj.send(@symbol_used_to_create_proc)
  end
end

10、使用 “method_missing”

class User
  def method_missing(_)
    hello
  end
end

user.i_am_a_lizard_king # prints "Hello, Gregory!"
user.i_can_do_everything # prints "Hello, Gregory!"

这种方式有点牵强,其实我使用的仍然是标准的方法,但我认为值得在此一提。

method_missing 方法会在对象收到未定义方法的调用时执行。它是一个非常强大的方法,是保障 Ruby 灵活性的基础之一,但是它也有可能引发很多不易被察觉的bug(以及一些性能问题),因此请谨慎使用。

11、使用 “eval”

eval("user.hello")

这种方式也有点牵强,因为我使用的仍然是标准的调用语法,但是它的工作原理有很大不同。eval 将该字符串传递给 Ruby 的解析器和解释器,就好像是我写的代码的一部分,然后执行该代码。在代码中千万不要使用这种写法,尤其是在允许用户将某些值传递给应用程序的情况下。

12、使用 "source" 和 "instance_eval"

require 'method_source' # external gem

method_source = user.method(:hello).source
method_body =method_source.split("\n")[1...-1].join(";")
user.instance_eval(method_source)

这是最后一个,稍微有点放飞自我,所以解释也有点长。这种方式需要依赖一个外部的包 method_source, 但这只是因为不用这个包的话,我就需要花费大量时间来编写这些代码(但都是 Ruby 代码,不需要借助魔法!)。下面我来解释一下其中的工作原理:

user.method(:hello).source 将以字符串的形式返回方法的源代码。其输出是整段代码(包括空格):

  def hello
    puts "Hello,#{@name}!"
  end

method_source 包是如何实现的?Ruby中的 Method 类拥有一个 source_location函数,该函数可以返回方法源代码的位置:文件以及方法开始处的行号。接下来,method_source 会打开这个文件,找到相应的行,找到 end (代表方法的结束),然后返回开头与结束之间的代码。

现在,我拥有了方法的完整代码,接下来我需要删除方法的定义和 end。在上述示例中,我只需要删除第一行和最后一行,但如果方法只有一行,那么就需要一些改动。第二行的输出是一个字符串,值为:puts"Hello, #{@name}!"。

最后,我将这个字符串传递给 user 对象的 instance_eval。instance_eval 的工作原理类似于 eval,但它执行代码的作用域不同。如果我调用 eval,则它将在整个文件的作用域上执行代码,但其中不包含@name 变量的定义。将其传递给 instance_eval,可以确保它使用正确的值。

还有其它方法吗?

 

以上只是一个有趣的小实验。我相信 Ruby 还有很多调用方法的方式,因为这是一款强大又非常灵活的语言。你知道其它方法吗?请在下面留言!

原文:https://www.notonlycode.org/12-ways-to-call-a-method-in-ruby/

本文为 CSDN 翻译,转载请注明来源出处。


更多精彩推荐
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值