Ruby中的迭代器详解
D瓜哥最近想做一个网站,另外,老早就有学习一门动态语言的想法,满足着两个条件的编程语言中,Ruby、Python是最合适的两种语言。现在Ruby on Rails如日中天,光芒万丈!所以,就选定了Ruby,从零开始学习。
前天看了Ruby的迭代器,对于我这个只学过Java、C/C++等的人来说,绝对是眼前一亮的感觉!而且是光彩夺目:没想到迭代器还可以这么玩,太简练太方便而且特别强大!然后,D瓜哥就迫不及待的想写一篇文章给大家介绍介绍Ruby的迭代器!
迭代器简介
先简单介绍一下迭代器。
1.一个Ruby迭代器就是一个简单的能接收代码块的方法(比如each这个方法就是一个迭代器)。特征:如果一个方法里包含了yield调用,那这个方法肯定是迭代器;
2.迭代器方法和块之间有如下传递关系:块被当成一个特殊参数传给迭代器方法,而迭代器方法内部在使用yield调用代码块时可将参数值传入块;
3.实际上,迭代器的功能就是一种回调!迭代器方法所属的类只负责遍历需要遍历的元素,而对元素所做的处理则通过回调代码块来实现;
4.Ruby中的容器对象(如数组、Range和Hash对象等)都包含了两个简单的迭代器,分别是each和collect。each可以认为是最简单的迭代器,它会对集合的每个元素调用块。 collect,将容器中的元素传递给一个块,在块中处理后返回一个包含处理结果的新数组;
迭代器详解
Ruby中的迭代器可以说五花八门,下面我们从字符串、数字、数组、Map、文件、目录等几个方面来简单介绍一下Ruby的迭代器。
字符串迭代器
在Java中,字符串类型的数据没有迭代器。所以,如果需要“遍历”字符串,需要将字符串做一些其他处理才行。但是,在Ruby中就有。下面,我们通过代码来演示一下:
str = "abc"
str.each_byte {|c| printf ">%c", c}; #
# 输出如下:(为了和代码区别,D瓜哥在输出前面人为地加了#。)
# 以下的输出展示,处理方式相同。
#>a>b>c
each_byte 是字串中用于处理每个字节的迭代器。每个字节都会代入块参数 c 中。
Ruby中,不仅有用于字节的迭代器,还有用于每行的迭代器。示例如下:
str = "abc\nefg\nhijk"
str.each_line{|l| print l}
# 输出如下:
#abc
#efg
#hijk
怎么样,是不是被Ruby简练但强大的迭代器所折服?!好戏还在后面,接着向下看。
数字迭代器
在Ruby中,“一切皆为对象”,甚至数字也是对象。这点和Java不一样。所以,对字的迭代器,对于我这个Java程序猿也是闻所未闻。让我们写两个示例,管窥一二。
第一个场景:对某段代码进行N(比如5)次操作。在Java中,需要写个循环,但是在Ruby中,只需要调用一下times方法即可。代码如下:
5.times {print "I love http://www.jb51.net/ \n"} # 真的就这么简单
# 输出如下:
#I love http://www.jb51.net/
#I love http://www.jb51.net/
#I love http://www.jb51.net/
#I love http://www.jb51.net/
#I love http://www.jb51.net/
第二个场景:求1到5的数字之和。这个也特别简单:
sum = 0
(1..5).each {|i| sum += i}
print "Sum="+sum.to_s
如果使用upto函数,还可以这样写:
sum = 0
1.upto(5) {|x| sum += x }
print "Sum="+sum.to_s
有时,我们的步进不一定是1,可能是2,例如奇数和。这种情况下,可以使用step函数。代码如下:
sum = 0
1.step(5, 2) do |y| # step函数第二个参数是步进。
sum += y
end
print "Sum="+sum.to_s
感觉有点扯远了。下面,我们讲讲数组相关的迭代器。
数组迭代器
见识过了数字相关的迭代器,我们再看看数组相关的迭代器。
第一个场景:便利数组并输出每个元素。直接上代码:
languages = ['Ruby', 'Javascript', 'Java']
languages.each_with_index do |lang, i|
puts "#{i}, I love #{lang}!"
end
#输出如下:
#0, I love Ruby!
#1, I love Javascript!
#2, I love Java!
有时,我们需要对数组的元素做出一个挑选,这时可以这样干:
# 找出符合条件的值
b = [1,2,3].find_all{ |x| x % 2 == 1 }
# b的值是 [1,3]
有时,我们需要删除数组中的某些值。这时:
# 迭代并根据条件刪除
a = [51, 101, 256]
a.delete_if {|x| x >= 100 }
# a的值是 [51]
再来一个例子:
# 找出最长字串find the longest word
longest = ["cat", "sheep", "bear"].inject do |memo,word|
( memo.length > word.length )? memo : word
end
puts longest
#输出如下:
#sheep
Map迭代器
在Java中,如果相对Map使用迭代器,必须把Map转化成List类型的容器才行。但是,在Ruby中,有直接针对Map的迭代器,很好很方便:
sum = 0
outcome = {"book1"=>1000, "book2"=>1000,"book3"=>4000}
outcome.each{|item, price|
sum += price
}
print "Sum="+sum.to_s
甚至,我们还可以这样:
sum = 0
outcome = {"book1"=>1000, "book2"=>1000,"book3"=>4000}
outcome.each{|pair|
sum += pair[1] # 读取值
}
print "Sum="+sum.to_s
这里说明一下:上述程序使用了pair[1]读取Map的值,如果要读取Map的键时则写成pair[0]。
如果需要输出Map的Key,可以这样:
outcome = {"book1"=>1000, "book2"=>1000,"book3"=>4000}
outcome.each_key do |k|
puts k
end
如果需要输出Map的value,则可以这样:
outcome = {"book1"=>1000, "book2"=>1000,"book3"=>4000}
outcome.each_value do |v|
puts v
end
文件迭代器
实在是没有想到,对于文件,Ruby也有迭代器可用。如下:
f = File.open("sample.txt")
f.each{|line|
print line
}
f.close
其实,我们可以使用代码块来进行同样的操作:
File.open("str.rb", "r") do |file|
file.each{|line|
print line
}
end
使用代码块,不需要手动close。这个推荐!
目录迭代器
很多时候,我们需要列出某个目录下的文件列表,设置对每个文件进行操作,这时也需要迭代器。Ruby也考虑到了:
Dir.foreach("c://") do |file| # 请根据自己的系统类型,做适当的修改
puts file
end
#输出太多,就不贴结果了。可以自己运行一下看看
结尾
通过上面的介绍可以看出,Java和Ruby相比,在迭代器方面简直是弱爆了!当然,D瓜哥刚刚开始学习Ruby,文中有不当甚至解释错误的地方,劳烦指出,D瓜哥会尽快改正的。