一、读书笔记
第三章 类、对象和变量
举个例子
class
def initialize(name, artist, duration)
@name = name
@artist = artist
@duration = duration
end
end
在Ruby程序中,initialize是一个特殊的方法,当你调用Song.new创建一个新的song对象的时候,Ruby首先分配一些内存来保存初始化的对象,然后调用对象initialize方法,并把调用new时所使用的参数传入该方法,这位你提供了机会编写设置对象的状态。
对song类来说,initialize方法接收3个参数,这些参数的作用同方法内的局部变量一样,因此它们遵循了局部变量命名的约定,即以小写字母开头。
每个对象都有都表示自己对应的歌曲,因此我们需要每个song对象带有自己的歌曲名、演唱者和时长。这意味着我们需要将这些值作为实例变量(instance variables)保存在对象中,对象内的所有方法都可以访问实例变量,每个对象都有实例变量的一份拷贝。
在Ruby中,实例变量就是一个由@符开头的名字。在我们的例子中,实例变量@name被参数name赋值,@artist被参数artist赋值,@duration被被参数duration赋值。
让我们测试一下这个绝妙的类。
song = Song.new("Bicyclops", "Fleck", 260)
song.inspect
返回
"#<Song:0x007faabdab80c8 @name=\"Bicyclops\", @artist=\"Fleck\", @duration=260>"
inspect方法(可以发送给任何对象),默认将对象的ID和实例变量格式化。
但是有的时候,inspect并不适用,我们可以用一个Ruby的标准消息(message)to_s,它可以发送给任何一个想要输出字符串的表示的对象。
song = Song.new("Bicylops", "Fleck", 260)
song.to_s
返回
"#<Song:0x007faabdab80c8>"
但是这并没有什么卵用,只是输出对象的ID而已,我们可以覆写(override)song中的to_s方法。我们使用#符号把3个实例变量插入到字符串中。
class Song
def to_s
"Song: #@name +++ #@artist (#@duration)"
end
end
song = Song.new("Bicyclops", "Fleck", 260)
song.to_s
棒极了,你学到了重要的一点,如何覆写to_s
3.1 让我们看一下继承和消息
继承允许你创建一个类,作为另一个类的精炼(refinement)和特化(specialization)。例如,在我们自动点唱机系统中,有了歌曲,被封装在song类中,但是我们想要有一个唱卡拉OK的功能,一首卡拉OK的歌曲和其他歌曲没什么区别,只是没有原唱,我们需要将歌词显示在屏幕上让唱K的人看到。
解决这个问题,就需要定义一个新类KaraokeSong,就是song加上歌词。
class KaraokeSong < Song
def initialize(name, artist, duration, lyrics)
super(name, artist, duration)
@lyrics = lyrics
end
end
song = KaraokeSong.new("My way", "Sinatra", 250, "And now ....")
song.to_s
返回
#<KaraokeSong:0x007faabe3206c0 @name="My way", @artist="Sinatra", @duration=250, @lyrics="And now ...."> # KaraokeSong.new
"Song: My way +++ Sinatra (250)" #song.to_s
song.to_s没有显示歌词呢?
这是因为在Ruby中,遇到方法调用song.to_s时,它并不知道从何处找到to_s方法,而是将判定推迟到直程序开始运行时,再进行,在那时Ruby查看song所属的类,如果该类实现了和消息名称相同的方法,就运行这个方法。
否则,Ruby就查看其父类中的方法,然后是祖父类,凡此以后追溯整个祖先链,如果最终它在祖先类中没有找到合适的方法,Ruby会产生一种特殊的行为,通常导致引发一个错误。
回到我们的例子,我们向song—即KaraokeSong类的一个对象——发送消息to_s,Ruby在KaraokeSong中没有找到要调用的方法to_s,然后解释器(interpreter)查看KaraokeSong的父类Song类,结果找到了to_s方法,这就是为什么输出了歌曲却没有歌词的原因——Song类并不知道歌词。
怎么解决呢?
class KaraokeSong
def to_s
"KS: #@name -- #@artist (#@duration) {#@lyrics}"
end
end
但是这样写就不够有扩展性,万一要修改父类中的duration怎么办?
假如决定把时长(duration)以毫秒保存,突然间,KaraokeSong开始报告非常荒谬的时长,这太恐怖了。
所以,合适的解法是这样的:
class KaraokeSong
def to_s
super + "[#@lyrics]"
end
end