25

12 种方式轻松实现 Ruby 调用

 3 years ago
source link: https://blog.csdn.net/dQCFKyQDXYm3F8rB0/article/details/108177463
Go to the source link to view the article. You can view the picture content, updated content and better typesetting reading experience. If the link is broken, please click the button below to view the snapshot at that time.

Zzeyee.jpg!mobile

7rIbqaF.jpg!mobile

作者 |  Gregory Witek

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

头图 | CSDN 下载自东方IC

出品 | CSDN(ID:CSDNnews)

以下为译文:

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

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

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

vmYnU3.jpg!mobile

准备工作

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

这个类叫做 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')

euUjAvZ.jpg!mobile

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,可以确保它使用正确的值。

NNfMrqz.jpg!mobile

还有其它方法吗?

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

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

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

nayQVbv.jpg!mobile

UZ3yUvj.jpg!mobile


About Joyk


Aggregate valuable and interesting links.
Joyk means Joy of geeK