加载中...

习题 45: 物件、类和从属关系


有一个重要的概念你需要弄明白,那就是 Class“类”和 Object“物件”的区别。问题在于,class 和 object 并没有真正的不同。它们其实是同样的东西,只是在不同的时间名字不同罢了。我用禅语来解释一下吧:

魚(Fish)和鮭魚(Salmon)有什麼區別?

这个问题有没有让你有点晕呢?说真的,坐下来想一分钟。我的意思是说,鱼和鲑鱼是不一样,不过它们其实也是一样的是不是?泥鳅是鱼的一种,所以说没什么不同,不过泥鳅又有些特别,它和别的种类的鱼的确不一样,比如鲑鱼和比目鱼就不一样。所以鲑鱼和鱼既相同又不同。怪了。

这个问题让人晕的原因是大部分人不会这样去思考问题,其实每个人都懂这一点,你无须去思考鱼和鲑鱼的区别,因为你知道它们之间的关系。你知道鲑鱼是鱼的一种,而且鱼还有别的种类,根本就没必要去思考这类问题。

让我们更进一步,假设你有一只水桶,里边有三条鲑鱼。假设你的好人卡多到没地方用,于是你给它们分别取名叫Frank,Joe,Mary。现在想想这个问题:

Mary 和鮭魚有什麼區別?

这个问题一样的奇怪,但比起鱼和鲑鱼的问题来还好点。你知道 Mary是一条鲑鱼,所以他并没什么不同,他只是鲑鱼的一个“实例(instance)”。Joe 和Frank 一样也是鲑鱼的实例。我的意思是说,它们是由鲑鱼创建出来的,而且代表着和鲑鱼一样的属性。

所以我们的思维方式是(你可能会有点不习惯):鱼是一个“类(class)”,鲑鱼是一个“类(class)”,而 Mary 是一个“物件(object)”。仔细想想,然后我再一点一点慢慢解释给你。

鱼是一个“类”,表示它不是一个真正的东西,而是一个用来描述具有同类属性的实例的概括性词汇。你有鳍?你有鳔?你住在水里?好吧那你就是一条鱼。

后来一个博士路过,看到你的水桶,于是告诉你:“小伙子,你这些鱼是鲑鱼。” 专家一出,真相即现。并且专家还定义了一个新的叫做​​“鲑鱼”的“类”,而这个“类”又有它特定的属性。长鼻子?红肉?体型大?住在海里或是干净新鲜的水里?吃起来味道不错?那你就是一条鲑鱼。

最后一个厨师过来了,他跟博士说:“非也非也,你看到的是鲑鱼,我看到的是Mary,而且我要把 Mary 淋上美味酱料做一道小菜。 ”于是你就有了一只叫做Mary 的鲑鱼的“实例(instance)”(鲑鱼也是鱼的一个“实例”),并且你使用了它(把它塞到你的胃里了),这样它就是一个​​“物件(object)”。

这会你应该了解了:Mary 是鲑鱼的成员,而鲑鱼又是鱼的成员。这里的关系式:物件属于某个类,而某个类又属于另一个类。

写成程式码是什么样子

这个概念有点诡异,不过实话说,你只要在建立和使用class的时候操心一下就可以了。我来给你两个区分 Class 和 Object的小技巧。

首先针对类和物件,你需要学会两个说法,“is-a(是啥)”和“has-a(有啥)”。“是啥”要用在谈论“两者以类的关系互相关联”的时候,而“有啥”要用在“两者无共同点,仅是互为参照”的时候。

接下来,通读这段程式码,将每一个注解为##??的位置标明他是“is-a”还是“has-a”的关系,并讲明白这个关系是什么。在程式码的开始我还举了几个例子,所以你只要写剩下的就可以了。

记住,“是啥”指的是鱼和鲑鱼的关系,而“有啥”指的是鲑鱼和烤肉架的关系。

## Animal is-a object (yes, sort of confusing) look at the extra credit
class Animal

end

## ??
class Dog < Animal

  def initialize(name)
    ## ??
    @name = name
  end

end

## ??
class Cat < Animal

  def initialize(name)
    ## ??
    @name = name
  end

end

## ??
class Person

  attr_accessor :pet

  def initialize(name)
    ## ??
    @name = name

    ## Person has-a pet of some kind
    @pet = nil
  end

end
## ??
class Employee < Person

  def initialize(name, salary)
    ## ?? hmm what is this strange magic?
    super(name)
    ## ??
    @salary = salary
  end

end

## ??
class Fish

end

## ??
class Salmon < Fish

end

## ??
class Halibut < Fish

end

## rover is-a Dog
rover = Dog.new("Rover")

## ??
satan = Cat.new("Satan")

## ??
mary = Person.new("Mary")

## ??
mary.pet = satan

## ??
frank = Employee.new("Frank", 120000)

## ??
frank.pet = rover

## ??
flipper = Fish.new

## ??
crouse = Salmon.new

## ??
harry = Halibut.new

加分习题

  1. 有没有办法把 Class 当作 Object 使用呢?
  2. 在习题中为 animals、fish、还有people 添加一些函式,让它们做一些事情。看看当函数在 Animal 这样的“基类(base class)”里和在 Dog 里有什么区别。
  3. 找些别人的程式码,理清里边的“是啥”和“有啥”的关系。
  4. 使用 Array 和 Hash 建立一些新的一对应多的“has-many”的关系。
  5. 你认为会有一种“has-many”的关系吗?阅读一下关于“多重继承(multiple inheritance)”的资料,然后尽量避免这种用法。

还没有评论.