继承允许你创建一个类,作为另一个类的精炼(refinement)和特化(specialization)。例如,在我们的自动点唱机系统中,有“歌曲”这一概念,被封装在Song类中,然后,随着市场的成长,我们需要提供卡拉OK的支持。一首卡拉OK歌曲和其他歌曲没什么两样(它只是没有主唱的音轨,对此我们不必关心)。不过,它还包括对于的一套歌词以及时间信息。当我们的自动点唱机在播放一首卡拉OK歌曲时,歌词应该随音乐滚动显示在点唱机前的屏幕上。

解决这个问题的一种方法是定义一个新的类KaraokeSong,就是Song加上歌词。

class KaraokeSong <Song

  def initialize(name,artist,duration,lyrics)

    super(name,artist,duration)

    @lyrics = lyrics

  end

end

类定义一行中的“< Song”告诉Ruby, KaraokeSong是Song 的子类(subclass).因此,这也意味着Song是KaraokeSong的超类(superclass)

song = KaraokeSong.new("My Way","Sinatra",255,"And now,the ...")

song.to_s        ->          *Song:My Way--Sinatra(225)*

调用to_s方法没有显示歌词

这和我们在向一个对象发送消息时,Ruby判定调用哪个方法的机制有关。在程序代码的初始解析(parse)期间,当Ruby遇到方法调用song.to_s时,它并不知道从何处找到to_s方法,而是将判定推迟直至程序开始运行时再运行。在那时,Ruby查看song所属的类。如果该类实现了和消息名称相同的方法,就运行这个方法。否则,Ruby就查看其父类中的方法,然后是祖父类,凡此以往追溯整个祖先链。如果最终它在祖先类中没有找到合适的方法,Ruby会产生一种特殊的行为,通常是导致引发一个错误。

让我们通过实现KaraokeSong#to_s来解决这个问题,你有许多方法可以完成它。让我们从最槽糕的方法开始,我们将to_s方法从Song类中拷贝出来并添加lyrics信息。

class KaraokeSong

 #...

 def to_s

   "KS: #@name--#@artist(#@duration){#@lyrics}"

  end

end

song = KaraokeSong.new("My Way", "Sinatra", 225,"And now,the...")

song.to_s     ->"KS: My Way--Sinatra(225){And now,the...}"

我们正确地显示了实例变量@lyrics的值。但使用这种方法,子类需要直接访问其祖先的实例变量。那么为什么这是实现to_s的一种糟糕方式呢?

答案与良好的编程风格有关(有时被称为解耦)。直接戳进父类的内部结构,并且显示地检验它的实例变量,会使得我们和父类的实现紧密地绑在一起。

我们通过让每个类处理其自身实现细节的方法来解决这个问题。当调用KaraokeSong#to_s时,我们调用其父类的to_s方法来得到歌曲的细节。然后,将歌词信息添加上去,并返回结果。这里使用的技巧是Ruby的关键字super。当你调用super而不使用参数时,Ruby向当前对象的父类发送一个消息,要求它调用子类中的同名方法。Ruby将我们原先调用方法时的参数传递给父类的方法。现在,我们可以实现改进后新的to_s方法。

class KaraokeSong <Song

  #Format ourselves as a string by appending

  #our lyrics to our parent's to_s value.

  def to_s

    super+"{#@lyrics}"

  end

end

song = KaraokeSong.new("My Way", "Sinatra" ,225, "And now,the...")

song.to_s      ->"Song:My Way--Sinatra(225){And now,the...}"

我们明确地告诉Ruby,KaraokeSong是Song的子类,但是我们并没有指定Song类本身的父类是什么。如果你在定义一个类时没有指定其父类,Ruby默认以Object类作为其父类。这意味着所有类的始祖都是Object,并且Object的实例方法对Ruby的所有对象都可用。

广告合作:本站广告合作请联系QQ:858582 申请时备注:广告合作(否则不回)
免责声明:本站资源来自互联网收集,仅供用于学习和交流,请遵循相关法律法规,本站一切资源不代表本站立场,如有侵权、后门、不妥请联系本站删除!

《魔兽世界》大逃杀!60人新游玩模式《强袭风暴》3月21日上线

暴雪近日发布了《魔兽世界》10.2.6 更新内容,新游玩模式《强袭风暴》即将于3月21 日在亚服上线,届时玩家将前往阿拉希高地展开一场 60 人大逃杀对战。

艾泽拉斯的冒险者已经征服了艾泽拉斯的大地及遥远的彼岸。他们在对抗世界上最致命的敌人时展现出过人的手腕,并且成功阻止终结宇宙等级的威胁。当他们在为即将于《魔兽世界》资料片《地心之战》中来袭的萨拉塔斯势力做战斗准备时,他们还需要在熟悉的阿拉希高地面对一个全新的敌人──那就是彼此。在《巨龙崛起》10.2.6 更新的《强袭风暴》中,玩家将会进入一个全新的海盗主题大逃杀式限时活动,其中包含极高的风险和史诗级的奖励。

《强袭风暴》不是普通的战场,作为一个独立于主游戏之外的活动,玩家可以用大逃杀的风格来体验《魔兽世界》,不分职业、不分装备(除了你在赛局中捡到的),光是技巧和战略的强弱之分就能决定出谁才是能坚持到最后的赢家。本次活动将会开放单人和双人模式,玩家在加入海盗主题的预赛大厅区域前,可以从强袭风暴角色画面新增好友。游玩游戏将可以累计名望轨迹,《巨龙崛起》和《魔兽世界:巫妖王之怒 经典版》的玩家都可以获得奖励。