首页 > 动态语言, 挨踢(IT) > 牛逼闪闪的Ruby迭代器

牛逼闪闪的Ruby迭代器

2013年1月7日 发表评论 阅读评论 2,277 人阅读    

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

# 输出如下:
#I love http://www.diguage.com/
#I love http://www.diguage.com/
#I love http://www.diguage.com/
#I love http://www.diguage.com/
#I love http://www.diguage.com/

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

参考资料

  1. Ruby程式語言入門
  2. 【学习笔记】RUBY 块和迭代器
  3. Ruby 使用手冊:迭代器 Iterators
  4. ruby迭代器基础


作 者: D瓜哥,https://www.diguage.com/
原文链接:https://wordpress.diguage.com/archives/86.html
版权声明:非特殊声明均为本站原创作品,转载时请注明作者和原文链接。

  1. 2013年1月10日23:00 | #1

    ruby的迭代器确实比较灵活简洁啊。

  2. 2013年3月11日13:22 | #2

    正因为有这个迭代器,ruby 里面才很少用到 for、while 之类的循环

  3. 2013年3月12日11:47 | #3

    233 但其实Ruby的迭代器和Java迭代器是两种不同的迭代方法,这个在GOF中是有描述的。Java允许用户直接操控迭代器对象本身,遍历操作对用户可见,仅仅屏蔽了内部实现,用户可以自由遍历。而Ruby更多的是方便,连遍历的操作都是屏蔽的,用户只要关心的元素的操作即可。但一旦我要求比较特殊的遍历方法或是顺序的时候,这种方法就有局限性了。

  4. ani di
    2013年8月28日14:24 | #4

    python也因迭代器/生成器而闻名,不过ruby的更丰富。
    推荐看看ruby作者写的《松本行宏的程序世界》

    • D瓜哥
      2013年8月28日14:32 | #5

      谢谢!这本书我看了大概两遍了,有些章节写的非常好,让我感觉醍醐灌顶!哈哈

  1. 2013年1月18日11:24 | #1
  2. 2013年3月7日10:53 | #2