今天要讲的是迭代器(iterator)。什么是迭代器呢?它的好处就是会一个一个地传回集合里的元素,让我们可以利用迭代方法做重复的事。

 

在Ruby里的collection集合里有Array数组和Hash哈希。

 

今天要介绍三种用在collection集合的迭代器分别叫each,map和collect,这也是常见的Ruby面试考题呢!

 

Ruby经典面试题目#14

Day14 each,map和collect比较?What's the difference between each,map and collect?

 

我们来使用这三种迭代器,在Array和Hash两种集合中各举几个生活化的例子,相信大家就比较容易暸解啰!

 

Array

Array#each

最近我在进行旅行存钱计划。我有三个银行帐户,NAB,CAN,和WESTPAC,开户金额分别为100,200,300。所以我写一个数组集合放入初始金额[100,200,300]。

 

假设我打算开始从本周开始在每个帐户存入50元(+50),可以在数组后加上.each方法:

 

[100,200,300].each {|n| puts n+50}

结果列出各个帐户的金额(注意:puts写在block大括号里):

 

150

250

350

经过.each方法作用之后,只会分别印出同一数组中各个元素的值,不会产生新数组。

 

Array#collect

以上案例,换成.collect试试看:(注意!puts写在前面)

 

puts [100,200,300].collect {|n| n+50}

.collect会帮我们把结果放入新的数组。结果印出(xujiaqing):

 

[150,250,350]

Array#map

同样的,.map方法也会帮我们产生新的数组。

 

puts [100,200,300].map {|n| n+50}

结果印出:

 

[150,250,350]

那.collect与.map又有什么不同?以及分别用在什么情况呢?这时候就要翻查Ruby手册里,关于.collect和.map的介绍了:

 

collect { |item| block }→new_ary Invokes the given block once for each element of self.Creates a new array containing the values returned by the block.

 

hmm…好像看不出有什么差异呢!

 

map { |obj| block }→array Returns a new array with the results of running block once for every element in enum.

 

近一步查询stackoverflow,果然有其他网友问过类似的问题。我们可以把.map当作是.collect的别名(map is an alias for collect),实务上比较常使用.map喔!

 

更多Array#map用法

现在把我的银行帐户数组存进account变数里,再用.inspect检查数组的值:

 

account = [100,200,300]

account.map {|n| n+50}

p account.inspect

结果印出

 

“[100,200,300]”

如果在.map后加上惊叹号.map!呢?

 

account.map!{|n| n+50}

p account.inspect

存进去原本的数组了。钱钱变多了!开心~~(加上!惊叹号的方法,通常代表小心!注意!的意思)

 

在这里的.map方法会让原本的物件数组被改变呢!

 

“[150,250,350]”

Hash

Hash哈希是一对key与value的集合。在刚刚的银行帐户例子里,我们可以把银行名称当作索引,存款数目当作值:

 

account = {“NAB”=> 100,“CAN”=> 200,“WEST”=> 300}

利用哈希来展现目前的信息(帐户名称、存款金额),这样就可读性更加清楚了。

 

Hash#each

现在我想要计算三个帐户加总共有多少钱,以.each的方式可写为:

 

mymoney = 0

account.each {|bankname,saving| mymoney += saving} #把索引和值列出

print“My Money: $”+ mymoney.to_s

或是

 

mymoney = 0 #设定初始值

account.each{|bank| mymoney += bank[1]} #依序加总bank集合里第二个元素bank[1]

print“My Money: $”+ mymoney.to_s

结果都会印出:

 

My Money: $ 600

Hash#map

在Hash里,把.each换成.map或是.collect:

 

mymoney = 0 #设定初始值

account.collect{|bank| mymoney += bank[1]} #依序加总bank集合里第二个元素bank[1]

print“My Money: $”+ mymoney.to_s

结果都是一样的:

 

My Money: $ 600

Hash#map结合Array#each与Array#map

现在要进阶到一个较为复杂的例子:hash里包含索引和值两部份,那我们可不可以把数组当作一种值,放在里面呢?

 

当然可以!

 

假设我的NAB银行下有2个子帐户,CAN银行下有3个子帐户,分别放入这些资产:

 

hash = {“NAB”=> [“Cash”,“Gold”],“CAN”=> [“Bitcoin”,“Litecoin”,“Ethereum”] }

利用hash.map会产生一个新的数组:(进一步了解说明,看stackoverflow的说明)

 

p hash.map {|n| n}

结果显示:

 

[[“NAB”,[“Cash”,“Gold”]],[“CAN”,[“Bitcoin”,“Litecoin”,“Ethereum”]]] #我有好多帐户!NAB下有2个,CAN下有3个

我想分别提取出银行:帐户名称的这一对信息,并且用逗号.join(“,”)隔开。

 

为了程序可读性,hash的索引命名为bank(银行名),值为account_arry(放了不同数目的子帐户数组)。在走account_arry.each展开数组迭代器时,每在集合里走完一个元素,就印出#{bank}:#{sub_account}

 

p hash.map {

|bank,account_arry| account_arry.each{

|sub_account|“#{bank}:#{sub_account}”}

}.join(“,”)

结果仅印出:

 

“Cash,Gold,Bitcoin,Litecoin,Ethereum”

奇怪,这不是我要的结果呀!我很希望帐户前面能显示出银行名称呢!

 

这是因为刚刚说过,arry.each会回传数组本身,在这个例子里,分别回传的是[“Cash”,“Gold”]和[“Bitcoin”,“Litecoin”,“Ethereum”]

 

改成.map试试看:

 

p hash.map {

|bank,account_arry| account_arry.map{

|sub_account|“#{bank}:#{sub_account}”}

}.join(“,”)

结果显示为:

 

“NAB: Cash,NAB: Gold,CAN: Bitcoin,CAN: Litecoin,CAN: Ethereum”

这是因为account_arry.map自动帮我们产生新的数组,放进bank与对应的sub_account并回传。

 

最后放个小小的比一比表格作为总结(expatvision):

 

each map / collect

Array方法Enumerable(列举)方法

回传Array本身产生新的Array并回传

祝福大家能顺利collect不同的资产,不管是有形的财富、还是无形的知识,最后都可以达成钱多多的心愿喔!:D