浅谈 Ruby 中的 block, proc, lambda, method object 的区别

简介:

当大家在百度中搜索“block proc lambda”的时候,会出来很多关于这几个概念之间区别的介绍,既然搜索结果中已经有了这些介绍,那为什么还要写这篇文章?

相信看过百度搜索结果中排名靠前的几篇文章的同学,都会发现其实这些文章并没有很好的说明他们之间区别是什么,大多只是介绍各自的用法,加上些许的区别,即使个别介绍了一些区别,也不够系统,不够深入。

正是基于上述原因,才酝酿了本文。本文以简单示例的方式,详细的介绍了它们之间的区别。相信您阅读完本文,一定会豁然开朗,并在今后的开发中准确并优雅的使用它们。

由于时间,个人能力水平等有限,本文难免有错误或缺失之处,欢迎不吝指出。

block & proc

在介绍它们的区别之前,我们先来看一段有关block的简单使用方法:

def use_yield                                                                                         
       yield
   end
   
   use_yield do
       puts 'use yield'
   end
   
   def use_block_call(&block)
       block.call
  end
 
  use_block do
      puts 'use block call'
  end

以上介绍了两种在函数中使用block的方式,第一种是通过yield来使用block,另外一种则是通过block.call来使用block。

proc全称为procedure,中文翻译为过程,步骤。关于block和proc的区别,我们先来看一个简单的示例。

def what_am_i(&block)                                                                                 
       block.class
   end
   puts what_am_i {}  # =< Proc

定义一个函数what_am_i并接受一个block,执行体中打印了block的类型,从执行的结果我们看到block的类型为proc,即说明block为proc的一种。

block和proc之间的区别主要有两个:一是proc可以重复使用,如果某个block将在多个地方被用到,则我们可以将其定义为proc。另外一个区别就是当某个函数需要执行多个闭包的时候,此时我们就无法使用block而只有使用proc或其他的闭包。

示例如下,在执行某个具体的操作的前后,调用了我们自己的proc。

def action(code)                                                                                      
       code[:before].call
       puts 'processing...'
       code[:after].call
   end
  
   action :before => Proc.new {puts 'before action'},
       :after => Proc.new {puts 'after action'}
 

proc & lambda

关于proc和lambda的区别,我们先来看一个简单的例子。

def args(code)                                                                                        
       code.call 'one','two'
   end
  
   args Proc.new { |a,b| puts a,b}
   args lambda {|a,b| puts a,b}

上述示例,第一眼看去发觉lambda和proc之间没什么区别或者很难发现它们的区别是什么。

接下来,我们对上述示例做一下简单的修改,我们将之前的接受两个参数修改为三个,看看会发生什么情况。

def args(code)
       code.call 'one','two'
   end
   
   args Proc.new { |a,b,c| puts a,b,c}
   args lambda {|a,b,c| puts a,b,c} # =< wrong number of arguments (2 for 3) (ArgumentError)

运行修改后的示例,发现lambda闭包出错了,出错的信息为,错误的参数个数。

至此,我们很快就发现了它们之间的一个区别是,lambda会检查参数的个数,而proc则不会,proc默认将缺失的参数视为nil,并继续执行代码,这是为什么呢?在本节的最后,我们会道出这其中的缘由。

在了解到它们的第一个区别之后,接下来我们再来看一个示例。

def proc_return•                                                                                      
       Proc.new {return 'Proc.new'}.call
       puts 'proc_return method finished'
   end
   
   def lambda_return
       lambda { return 'lambda'}.call
       puts 'lambda_return method finished'
   end
   
  puts proc_return # =< Proc.new
  puts lambda_return # =< lambda_return method finished

这个示例非常的简单,我们定义了两个函数proc_return以及lambda_return。这两个函数一个用proc实现,另外一个用lambda实现,执行体都只有一行代码,就是返回一段字符串。

执行的结果出乎我们的意料,proc并未执行return之后的代码,而lambda执行了return之后的代码。

综上所述,我们得出了proc和lambda的两个重要区别,一是lambda会进行参数个数的检查而proc则不会,另外lambda会执行return之后的代码而proc则不会。

为什么会出现上述情况,本质的原因在于,proc只是一段嵌入的代码片段而lambda则是匿名函数,正因为是匿名函数,所以会检查函数调用参数,并在函数调用结束之后,继续执行后面的代码,而proc由于是嵌入的一段代码片段,在执行完return语句后,就已经返回,所以不再执行之后的代码。
lambda & method object



   def greeting                                                                                          
       puts 'hello, method object'
   end
   
   method(:greeting).call
   
   lambda{puts 'hello, lambda'}.call

lambda和method object的用法基本一致,其唯一的区别在于lambda为匿名函数,而method object为命名函数。

总结

关于block,proc,lambda, method object这四者之间的区别可总结为以下:

block和proc本质上是一段嵌入的代码块,并非函数。而lambda和method object都是函数,只不过lambda是匿名函数,而method object为命名函数。

从本质上理解了它们的区别,我们在今后的开发中就会正确且优雅的运用它们。

文章转载自 开源中国社区 [http://www.oschina.net]

相关文章
|
10月前
|
Ruby
浅谈Ruby中的block, proc, lambda, method object的区别
浅谈Ruby中的block, proc, lambda, method object的区别
61 0
|
Ruby 容器
【Ruby on Rails全栈课程】2.7 块(Block)和迭代器
1、块(Block) 块是ruby 的一个独特特性,是一种可以和方法调用相关联的代码块。 是在花括号{}或者do…end之间的一组代码,一般我们在单行中block用花括号{},在多行中用do…end (1)块的格式
95 0
Block、委托、回调函数原理剖析(在Object C语境)——这样讲还不懂,根本不可能!
开篇:要想理解Block和委托,最快的方法是搞明白“回调函数”这个概念。 做为初级选手,我们把Block、委托、回调函数,视为同一原理的三种不同名称。也就是说,现在,我们把这三个名词当成一回事。在这篇文章内,Block就是回调函数,委托也是回调函数,不再作详细的区分了。
911 0
ruby的方法和block
在ruby中方法与block的合用 :~/ruby$ irbirb(main):001:0> def take_block(p1)irb(main):002:1>   if block_given?irb(main):003:2>      yield(p1)irb(main):004:2>   ...
614 0
ruby的lambda
irb(main):012:0> def n_times(thing)irb(main):013:1>    return lambda {|n| thing * n}irb(main):014:1> end=> nilirb(main):015:0> p1=n_times(23)=> #irb(main):016:0> p1.
506 0
|
Go Ruby
Ruby学习笔记-Block, Proc and Lambda
1.Block:   Ruby中的块就是由多行代码组成的一个代码块,通常可以把它认为是一个匿名方法,常用来迭代一个数组或范围(如each, times方法);语法格式如下:   {                       //code   }             OR   do...
691 0
|
Ruby
ruby学习笔记(10)-puts,p,print的区别
共同点:都是用来屏幕输出的。 不同点:puts 输出内容后,会自动换行(如果内容参数为空,则仅输出一个换行符号);另外如果内容参数中有转义符,输出时将先处理转义再输出p 基本与puts相同,但不会处理参数中的转义符号print 基本与puts相同,但输出内容后,不会自动在结尾加上换行符 s = ...
826 0
|
1月前
|
数据采集 Web App开发 数据处理
Ruby网络爬虫教程:从入门到精通下载图片
Ruby网络爬虫教程:从入门到精通下载图片