Day4 读书笔记&心得体会

一、读书笔记
第三章 类、对象和变量

举个例子

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 

返回

"#"

inspect方法(可以发送给任何对象),默认将对象的ID和实例变量格式化。
但是有的时候,inspect并不适用,我们可以用一个Ruby的标准消息(message)to_s,它可以发送给任何一个想要输出字符串的表示的对象。

song = Song.new("Bicylops", "Fleck", 260)
song.to_s

返回

"#"

但是这并没有什么卵用,只是输出对象的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.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

你可能感兴趣的:(Day4 读书笔记&心得体会)