也谈面向对象
OO的三大基础是封装、继承、多态。
这三者是有次序性的。没有封装就不可能有继承、没有继承就不可能有多态。
【封装(encapsulation)】
封装的目的是要将代码分割成很多模块(module)。每一个模块之间的关连性降到最低,这么一来比較不会产生“牵一发而动全身”的状况。减少相互依赖的程度,也等于是减少复杂度,能够让开发与维护更easy。
其实,没有人用“模块”一词来称呼封装的结果。而是称为“类”,把模块一词做更高阶的包装用途。因此我们如今应该将“类”视为封装的结果,把“模块”视为整个程序分割出来的很多片段。而在OO的世界。一般来说。一个程序有多个模块。一个模块内包括多个类。
封装是以数据为核心。将相关的数据放在一起,将会用到这些数据的函数也放进来。封装等于是将数据和函数放在一起。
【能见度(visibility)】
封装的目的既然是要“减少互相依赖的程度”。就牵涉到能见度的问题:这个类/方法/属性该不该暴露给别的模块、同一个模块的不同类、自己的派生类、友元类?这就是所谓的能见度。
我们当然希望尽可能减少能见度,这才干减少互相依赖的程度。也就是。别人不须要知道的,就不要让它知道,这就是所谓的信息隐藏。
最该被隐藏的是数据。有些人主张全部的数据一定都不能够直接被外部(包含派生类)訪问。
上面提到。封装将相关的数据和使用到这些数据的方法包成类。
最理想的状况是,让数据的能见度为最低,外面全然看不见。
留下的对外接口(Interface)仅仅剩下method。换句话说,每一个对象的Interface是一些方法的集合。全然没有数据。
但设定能见度不是一件easy的事。往往须要深思熟虑。能见度设定得太宽,造成信息隐藏效果不佳,可能会带来相当多负面的效果。能见度设定得太紧,造成效率变差、扩充程度变差。
【继承】
被继承的对象称为基类(base)或父类(parent),继承者称为派生类(derived)或子类(child)。
继承的目的。是要达到“代码复用”或“接口复用”。而继承的手段,就是“扩充”或“改动”。这是继承的重点。
继承所导致的代码复用,是指派生类能自己主动沿袭基类的全部代码,好让你能够不用写太多代码。仅仅须要略微扩充或改动,就能符合你的需求。
扩充指的是:定义新的方法(Method)。改动指的是:针对基类中的某方法又一次定义其行为。
(PS:代码复用,通过“继承”(Inheritance)和“组合”(Composition)都能够实现。且如今很多其它的设计是推荐使用“组合”,组合是动态的,有更好的灵活性。继承很多其它的是为了多态。)
继承所导致的接口复用。是在为OO的下一个阶段(也就是多态)作准备。接口复用,搭配方法的改动。就形成了多态。
假设你不想复用代码。也不想复用接口,或者说你不进行扩充、也不进行改动,那么透过继承产生派生类,差点儿是没有意义的。
唯一的一个小小的意义是,派生类和基类两者是不同的类,你能够在程序中根据这一点做推断,做不同的行为。可是这是一种琐细的程序技巧(RTTI),和OO无关。并且OO也不鼓舞你这么做。
在考虑使用继承时。有一点须要注意,那就是两个类之间的关系应该是“is-a”关系。比如。Employee 是一个人,Manager 也是一个人。因此这两个类都能够继承 Person 类。
可是 Leg 类却不能继承 Person 类。由于腿并非一个人。
【多重继承与接口】
单一继承指的是,仅仅有一个基类;多重继承指的是,具有多个基类。
多重继承可能会造成不知继承的方法是来自那个基类或祖先类的困扰。 C++要求编程员要主动指明继承的方法来自何处。
所以有些语言往往禁止多重继承(比如Java等)。
从Java開始,多数的语言使用Interface来解决多重继承的问题。但Interface仅仅能让你继承到Interface,无法继承到代码(Interface不带代码)。因此。假设你在Java中继承多个Interface。你必须亲自己定义全部Interface的每一个方法,也就是说,你必须写很多代码。但假设是在C++中,你能够继承很多类。不须要再定义这些方法。
设计继承时,必须先考虑接口是否共享,再考虑代码是否共享,再考虑分类。可是一般的程序猿。反倒会先考虑分类和代码复用。而忽略了"接口复用"是当中最重要的事。
【多态Polymorphism】
多态是同意你将父对象设置成为和一个或很多其它的他的子对象相等的技术,赋值之后,父对象就能够依据当前赋值给它的子对象的特性以不同的方式运作。简单的说,就是一句话:同意将子类类型的指针赋值给父类类型的指针。
多态让人认为对象能够以多种面貌出现。当一个对象具有不同的型态。就有可能会引发多态机制。
一个对象为何为有不同的型态?这是由于继承而来。对象能够扮演全部祖先类的角色。比如当某对象的类是派生类,当此对象被“转型”成基类之后。此对象就具有两种不同的类,“实际类”是派生类,“形式类”是基类。此时调用此对象的方法m,会运行到的是基类定义的方法m?还是派生类定义(改动)的方法m?
答案是实际类的方法。也就是派生类定义的方法m。所以所谓的多态就是:无论形式类是什么,一定会运行到实际类的方法。
你可能会认为疑惑。为何当初要将对象转型为祖先类。导致“形式类”(宣告类)和“实际类”(定义类)不一样?
由于我们想用一个东西(基类)来表示多个东西(派生类)。
更重要的是,良好的设计是面向接口编程。
我们在基类定义了接口。在派生类实现,不同的派生类能够有不同的实现。我们能够让接口的使用者不变(使用对象的“形式类”)。而改变其“实际类”,改变类的行为。这样能够使程序易于改动和扩展。
[比方,框架。]
类的方法,能够分成虚拟(Virtual)与非虚拟两种。仅仅有虚拟方法才干搭配多态机制使用。
假设是非虚拟方法。则会运行到形式类(而非实际类)的方法。由于多态没有发挥作用。
关于虚拟。每一个语言有不同的作法。 Java强调动态。所以默认是虚拟。C++注重效率,所以默认不是虚拟。
切记:「接口复用」比「代码复用」更重要。这是由于多态的缘故,多态才是OO的终极目的。
[话外:关于重载与覆盖]
覆盖。是指子类又一次定义父类的虚函数的做法。
(我不喜欢“覆盖”这个名字,easy把人搞晕)
重载。是指同意存在多个同名函数,而这些函数的參数表不同
重载的实现是:编译器依据函数不同的參数表。对同名函数的名称做修饰,然后这些同名函数就成了不同的函数(至少对于编译器来说是这种)。
对于这两个函数的调用。在编译器间就已经确定了。是静态的(记住:是静态)。
也就是说。它们的地址在编译期就绑定了(早绑定),因此,重载和多态无关!
真正和多态相关的是“覆盖”。当子类又一次定义了父类的虚函数后。父类指针依据赋给它的不同的子类指针,动态(记住:是动态。)的调用属于子类的该函数,这种函数调用在编译期间是无法确定的(调用的子类的虚函数的地址无法给出)。因此,这种函数地址是在执行期绑定的(晚邦定)。
结论就是:重载仅仅是一种语言特性,与多态无关。与面向对象也无关!
引用一句Bruce Eckel的话:“不要犯傻。假设它不是晚邦定,它就不是多态。
”
[注:本段改编自(http://www.cnblogs.com/clongge/archive/2008/07/09/1239076.html)]
【框架】
为了方便软件的开发。很多软件厂商都会提供应用框架(ApplicationFramework)。现今流行的框架相当多,比如:.NET Framework等。
有了框架。我们最终能够享受到OO的优点。反复利用别人写好的代码。不用一切自己重头写。
框架厂商先将一大部分的程序先写好,程序猿仅仅须要“利用继承来做改动”,就能套用整个框架。为了要让你改动的部分可以确实被运行到(而不是运行到框架本身的方法)。所以这些同意改动的方法都是定义成虚拟的。
由于程序猿“利用继承来做改动”所以产生了派生类和又一次定义的方法。
框架比这个派生类更早被定义,当然不认识这个派生类,所以框架内都是以此派生类的祖先类为“形式上”的处理对象(处理接口)。
当此派生类对象被传入框架中,就会被自己主动转型成为祖先类,因此产生“形式类”和“实际类”的差异。正由于这种类差异,加上次类有又一次定义方法,所以多态机制出现了。
【乱谈面向对象】
面向过程编程和面向对象编程,是两种不同的世界观。
从面向过程来看。解决一个问题是分多个步骤的流程。而从面向对象来看。解决一个问题靠的是相关类的协作。
面向对象是对现实世界直观的模拟,面向对象程序中核心的问题是,类与类之间的关系。
不同的事物有不同的属性。且有不同的能力(类的方法)。对象与对象之间传递消息。依据收到的信息。做出不同的反应。由此协作来解决这个问题。
类是有同样属性的对象的一种抽象,而基类之于派生类。是在一些派生类中提取共性而成的基类。
是一种更高层次的抽象。
抽象,是人类理解改造这个世界的方法。人类总是试图用最简单的理论解释很多其它的现象。
各个学科和领域也都有对应的理论(比方热力学方程、电磁学方程),它们是对某些范围的事物的一种抽象。牛顿的万有引力。是对全部事物的一种抽象,他提取了万物的共性。这是最高层次的抽象。
(如今比較热门的“弦论”有望成为包罗全部理论的大一统理论。它或许是人类对宇宙的终极抽象吧。^_^)
[注:以上内容有參考网上的。有參考书中的,为平日所搜集,今日之整理,外加自己的一些乱谈。作为一个0基础菜鸟,有些地方的理解可能是有问题的。希望大家不吝指正。]