详解Ruby中的代码块及其参数传递

一,块的声明 
  块的声明在函数调用之后,用{..}括起来,或do..end封装。{}一般用在单行语句上,do..end用在多行语句上。

(1..4).each{|v| print "#{v} "} #输出1 2 3 4

块可以带参数,与函数参数不同,块参数用||封装,当然,可以带多个参数。这些参数怎么定义,实际上是在函数内部定义好的,后面会讲到。

二,块内变量的访问 
  块内可以访问块外的变量,也就是块外的变量在块内是可见的,如

sum = 0
(1..5).each do |v|
  name = 'smile' #name属于块内变量,其可视范围只能在块内。假设块外没有相同名称的变量.
  sum += v #sum在块内可见
end
p sum #输出15,sum已改变。
p name #Error! name不可访问。

正因块内可以块外的变量所以可能不小心修改了一些外部变量,这是我们不希望的。幸运的是Ruby1.9版本后,提供了一种安全的方式声明块内变量,在块参数后面加";",块内变量放在";"之后.

name = 'outside'
sum = 0
(1..5).each do |v;name| #name在";"之后,可以声明多个变量,用逗号隔开
  name = 'inside' #name属于块内变量,其可视范围只能在块内.假设块外没有相同名称的变量。
  sum += v #sum在块内可访问
end
p sum #输出15,sum已改变。
p name #输出outside,没有变。

三,yield语句 
  看这里,可能还不是很明白,函数是如何调用块的。现在就来介绍块的调用,关键是yield语句。在函数体中,如果用yield,函数会调用函数的块。

def threeTime
  yield
  yield
  yield
end
threeTime{p 'Hello world!'}

输出三行Hello world!,是不是很简单呢。现在应该明白了吧,是yield调用的块。
块的参数是怎么回事呢?估计你已经想到了,就是yield的参数,跟一般函数一样yield可以带参数的。看例子

def takeBlock(p1)
 if block_given? # 判断是否有块,如果在yield时,没有声明块,会出错,所以在这里作判断会好点。
  yield(p1) #把p1传给块参数,既下面块声明中的s
 else
  p1
 end
endie 

takeBlock("no block")  #输出"no block"
takeBlock("no block") { |s| s.sub(/no /, '') } #输出"block"

既然yield能传参数给块,反过来,块能不能传值给yield呢?答案是肯定的。块中最后一句语句的值会自动传给yield。请看示例

def nTime
 i = yield #第一次调用时,返回块的值
 (0..i).each {|v| yield(v)} # 此处yield也可以放在块中
end
nTime do |v|
 print "#{v} " if v
 9 #yield调用时返回的数
end
#输出1 2 3 4 5 6 7 8 9

当然上例只是拿来做例子,实际上没有人会这样定义,更好的定义如下:

def nTime(n)
 (0..n).each {|v| yield(v)}
end
nTime(9) do |v|
 print "#{v} "
end

我们来看下Array中的find实现

class Array
 def find
  for i in 0...size
   value = self[i]
   return value if yield(value)
  end
  return nil
 end
end
[1, 3, 5, 7, 9].find {|v| v > 5 } #实现查找第一个大于5的数,输出7。

因为块的出现,Ruby中少了许多for语句,代码看上去更人性化,写代码不再是枯燥的事,而是一种享受。

四,传递块的另一种方式

def fun #不带参数的
 yield
end
proc = ->{p 'haha'} 

fun &proc
#####
def fun2(x) #带参数的
 yield x
end
proc2 = ->(x){p x}
fun2 1,&proc2

五,instance_eval()和instance_exec()
在Ruby中,提供了一个非常酷的特性,可以通过使用Objec#instance_eval(), Objec#instance_exec()方法插入一个代码块,做一个的对象上下文探针(Context Proble),深入到对象中的代码片段,对其进行操作。有了这个特性以后,就可以很轻松的测试对象的行为,查看对象的当前状态。

class MyClass
 def initialize
  @v = 1;
 end
end
obj = MyClass.new
obj.instance_eval do
 puts self       # => #<MyClass:0x007fbb2d0299b0>
 puts @v        # => 1
end
obj.instance_exec(5) { |x| puts x * @v } # => 5
(0)

相关推荐

  • 深入理解Ruby中的代码块block特性

    block是什么? 在Ruby中,block并不罕见.官方对block的定义是"一段被包裹着的代码".当然,我觉得这样的解释不会让你变的更明白. 对block的一种更简单的描述是"一个block就是一段存储在一个变量中的代码,它和其他的对象一样,可以被随时的运行" 然后,咱们通过看一些代码,之后再把这些代码重构成Ruby中的block形式.通过代码来实际的感受,更加直观. 比如,对两个数做加法? puts 5 + 6 # => 11 嗯,这样写是可以的.但是,

  • 详解Ruby中的代码块对象Proc

    Proc对象 Proc是由块转换来的对象.创建一个Proc共有四种方法,分别是: 示例代码 # 法一 inc = Proc.new { | x | x + 1} inc.call(2) #=> 3 # 法二 inc = lambda {| x | x + 1 } inc.call(2) #=> 3 # 法三 inc = ->(x) { x + 1} inc.call(2) #=> 3 # 法四 inc = proc {|x| x + 1 } inc.call(2) #=> 3

  • Ruby中的block代码块学习教程

    1.什么是代码块 在Ruby中,{}或do...end之间的代码是一个代码块.代码块只能出现在一个方法的后边,它紧接在方法最后一个参数的同一行上,由yield关键字调用.例如: [1,2,3,4,5].each { |i| puts i } [1,2,3,4,5].each do |i| puts i end 块变量:以yield关键字调用block也可以传递参数,block中竖线(|)之间给出的参数名用于接收来自yield的参数. 竖线之间(如上例中的 | i |)的变量被称作块变量,作用和一

  • 详解Ruby中的代码块及其参数传递

    一,块的声明    块的声明在函数调用之后,用{..}括起来,或do..end封装.{}一般用在单行语句上,do..end用在多行语句上. (1..4).each{|v| print "#{v} "} #输出1 2 3 4 块可以带参数,与函数参数不同,块参数用||封装,当然,可以带多个参数.这些参数怎么定义,实际上是在函数内部定义好的,后面会讲到. 二,块内变量的访问    块内可以访问块外的变量,也就是块外的变量在块内是可见的,如 sum = 0 (1..5).each do |v

  • 详解Ruby中的instance_eval方法及其与class_eval的对比

    instance_eval方法 这个BasicObject#instance_eval有点类似JS中的bind方法,不同的时,bind是将this传入到对象中,而instance_eval则是将代码块(上下文探针Context Probe)传入到指定的对象中,一个是传对象,一个是传执行体.通过这种方式就可以在instance_eval中的代码块里访问到调用者对象中的变量. 示例代码 class MyClass def initialize @v = 1 end end obj = MyClass

  • 详解Ruby中的块的知识

    语法: block_name{ statement1 statement2 .......... } 在这里,将学习如何通过使用一个简单的 yield 语句调用块.还将学习使用yield语句具有参数调用块.将检查的示例代码,这两种类型的 yield 语句. yield 语句: 让我们来看看在yield语句的一个例子: #!/usr/bin/ruby def test puts "You are in the method" yield puts "You are again

  • 详解Ruby中的循环语句的用法

    Ruby 中的循环用于执行相同的代码块若干次.本章节将详细介绍 Ruby 支持的所有循环语句. Ruby while 语句 语法 while conditional [do] code end 当 conditional 为真时,执行 code.while 循环的 conditional 通过保留字 do.一个换行符.反斜线 \ 或一个分号 ; ,来与 code 分离开. 实例 #!/usr/bin/ruby $i = 0 $num = 5 while $i < $num do puts("

  • 详解Ruby中的异常

    异常和执行总是被联系在一起.如果您打开一个不存在的文件,且没有恰当地处理这种情况,那么您的程序则被认为是低质量的. 如果异常发生,则程序停止.异常用于处理各种类型的错误,这些错误可能在程序执行期间发生,所以要采取适当的行动,而不至于让程序完全停止. Ruby 提供了一个完美的处理异常的机制.我们可以在 begin/end 块中附上可能抛出异常的代码,并使用 rescue 子句告诉 Ruby 完美要处理的异常类型. 语法 begin # - rescue OneTypeOfException #

  • 详解Ruby中的单件方法和单件类

    单件方法 Ruby允许给单个对象增加方法,这种只针对单个对象生效的方法,称为单件方法 示例代码 str = "just a regular string" def str.title? self.upcase == self end str.title? # => false str.methods.grep(/title?/) # => [:title?] str.singleton_methods #=> [:title?] str.class # => S

  • 详解Ruby中的方法概念

    Ruby方法跟其他编程语言中的函数非常相似, Ruby方法用于捆绑到一个单元中的一个或多个重复的语句. 方法名称应以小写字母开始.如果一个方法的名称以大写字母开始,Ruby可能会认为这是一个常数,因此可以正确解析调用. 方法应该定义Ruby的之前调用他们,否则会引发一个异常未定义的方法调用. 语法: def method_name [( [arg [= default]]...[, * arg [, &expr ]])] expr.. end 所以,可以定义一个简单的方法如下: def meth

  • 详解vscode中vue代码颜色插件

    vue提示插件[Vscode] 编者寄语:vscode的确是前端开发中很好的工具,安装颜色插件,从视觉上是美的享受.曾经的我遇到了vscode代码全是灰色,黑色的困惑,后来整理找到方法,整理这篇,以下高亮插件,建议大家都安装了. 在VSCode Marketplace 搜素Vue 出现关于语法高亮的插件有 vue,vue-beautify,vue-color,VueHelper,vertur等等.比较了下载数量可以了解到,vetur 是目前比较好的语法高亮插件,我们来安装一下吧. vscode提

随机推荐