Ruby中的迭代器详解

Tags: 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 / \n"} # 真的就这么简单

# 输出如下:
#I love /
#I love /
#I love /
#I love /
#I love /

第二个场景:求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瓜哥会尽快改正的。

本文链接:http://www.4byte.cn/learning/1347/ruby-zhong-de-die-dai-qi-xiang-jie.html



相关文章