12 种方式轻松实现 Ruby 调用

2020 年 8 月 22 日 CSDN
作者 | 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 翻译,转载请注明来源出处。


更多精彩推荐

           
           
             
实用小技能 | 用 Word 和 Excel 自制一个题库自判断答题系统!
中国数据库产业的“高地战事”
融资 2000 万美元后,他竟将核心代码全开源,这……能行吗?
Get了!用Python制作数据预测集成工具 | 附代码
学会这10大高性能开发技术,轻松躲过裁员名单!
小心!你可能玩了假的DeFi
点分享
点点赞
点在看
登录查看更多
0

相关内容

Ruby 是一种面向对象、命令式、函数式、动态的通用编程语言。
【2020新书】C语言编程傻瓜式入门,第二版,464页pdf
专知会员服务
60+阅读 · 2020年10月15日
【经典书】C语言傻瓜式入门(第二版),411页pdf
专知会员服务
51+阅读 · 2020年8月16日
【经典书】算法C语言实现,Algorithms in C. 672页pdf
专知会员服务
80+阅读 · 2020年8月13日
【Java实现遗传算法】162页pdf,Genetic Algorithms in Java Basics
专知会员服务
42+阅读 · 2020年7月19日
《深度学习》圣经花书的数学推导、原理与Python代码实现
5大必知的图算法,附Python代码实现
AI100
4+阅读 · 2019年9月10日
100行Python代码,轻松搞定神经网络
大数据文摘
4+阅读 · 2019年5月2日
你真的会正确地调试 TensorFlow 代码吗?
数据库开发
7+阅读 · 2019年3月18日
如何编写完美的 Python 命令行程序?
CSDN
5+阅读 · 2019年1月19日
Python3爬虫之入门和正则表达式
全球人工智能
7+阅读 · 2017年10月9日
代码这样写不止于优雅(Python版)
数说工作室
4+阅读 · 2017年7月17日
高效使用 Python 可视化工具 Matplotlib
Python开发者
8+阅读 · 2017年7月3日
Learning Recommender Systems from Multi-Behavior Data
Arxiv
7+阅读 · 2018年11月29日
Angular-Based Word Meta-Embedding Learning
Arxiv
3+阅读 · 2018年8月13日
Arxiv
8+阅读 · 2018年1月25日
VIP会员
相关资讯
5大必知的图算法,附Python代码实现
AI100
4+阅读 · 2019年9月10日
100行Python代码,轻松搞定神经网络
大数据文摘
4+阅读 · 2019年5月2日
你真的会正确地调试 TensorFlow 代码吗?
数据库开发
7+阅读 · 2019年3月18日
如何编写完美的 Python 命令行程序?
CSDN
5+阅读 · 2019年1月19日
Python3爬虫之入门和正则表达式
全球人工智能
7+阅读 · 2017年10月9日
代码这样写不止于优雅(Python版)
数说工作室
4+阅读 · 2017年7月17日
高效使用 Python 可视化工具 Matplotlib
Python开发者
8+阅读 · 2017年7月3日
Top
微信扫码咨询专知VIP会员