一,块的声明 
  块的声明在函数调用之后,用{..}括起来,或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调用的块。
块的参数是怎么回事呢"htmlcode">

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

  既然yield能传参数给块,反过来,块能不能传值给yield呢"htmlcode">

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 

广告合作:本站广告合作请联系QQ:858582 申请时备注:广告合作(否则不回)
免责声明:本站资源来自互联网收集,仅供用于学习和交流,请遵循相关法律法规,本站一切资源不代表本站立场,如有侵权、后门、不妥请联系本站删除!

《魔兽世界》大逃杀!60人新游玩模式《强袭风暴》3月21日上线

暴雪近日发布了《魔兽世界》10.2.6 更新内容,新游玩模式《强袭风暴》即将于3月21 日在亚服上线,届时玩家将前往阿拉希高地展开一场 60 人大逃杀对战。

艾泽拉斯的冒险者已经征服了艾泽拉斯的大地及遥远的彼岸。他们在对抗世界上最致命的敌人时展现出过人的手腕,并且成功阻止终结宇宙等级的威胁。当他们在为即将于《魔兽世界》资料片《地心之战》中来袭的萨拉塔斯势力做战斗准备时,他们还需要在熟悉的阿拉希高地面对一个全新的敌人──那就是彼此。在《巨龙崛起》10.2.6 更新的《强袭风暴》中,玩家将会进入一个全新的海盗主题大逃杀式限时活动,其中包含极高的风险和史诗级的奖励。

《强袭风暴》不是普通的战场,作为一个独立于主游戏之外的活动,玩家可以用大逃杀的风格来体验《魔兽世界》,不分职业、不分装备(除了你在赛局中捡到的),光是技巧和战略的强弱之分就能决定出谁才是能坚持到最后的赢家。本次活动将会开放单人和双人模式,玩家在加入海盗主题的预赛大厅区域前,可以从强袭风暴角色画面新增好友。游玩游戏将可以累计名望轨迹,《巨龙崛起》和《魔兽世界:巫妖王之怒 经典版》的玩家都可以获得奖励。