也谈面向对象

OO的三大基础是封装、继承、多态。

这三者是有次序性的,没有封装就不可能有继承、没有继承就不可能有多态。

【封装(encapsulation)】

封装的目的是要将代码切割成许多模块(module),每个模块之间的关连性降到最低,这么一来比较不会产生“牵一发而动全身”的状况,降低相互依赖的程度,也等于是降低复杂度,可以让开发与维护更容易。

事实上,没有人用“模块”一词来称呼封装的结果,而是称为“类”,把模块一词做更高阶的包装用途。因此我们现在应该将“类”视为封装的结果,把“模块”视为整个程序切割出来的许多片段。而在OO的世界,一般来说,一个程序有多个模块,一个模块内包含多个类。

封装是以数据为核心,将相关的数据放在一起,将会用到这些数据的函数也放进来。封装等于是将数据和函数放在一起。

【能见度(visibility)】

封装的目的既然是要“降低互相依赖的程度”,就牵涉到能见度的问题:这个类/方法/属性该不该暴露给别的模块、同一个模块的不同类、自己的派生类、友元类?这就是所谓的能见度。

我们当然希望尽可能降低能见度,这才能降低互相依赖的程度。也就是,别人不需要知道的,就不要让它知道,这就是所谓的信息隐藏。

最该被隐藏的是数据。有些人主张所有的数据一定都不可以直接被外部(包括派生类)访问。

上面提到,封装将相关的数据和使用到这些数据的方法包成类。最理想的状况是,让数据的能见度为最低,外面完全看不见。留下的对外接口(Interface)只剩下method。换句话说,每个对象的Interface是一些方法的集合,完全没有数据。

但设定能见度不是一件容易的事,往往需要深思熟虑。能见度设定得太宽,造成信息隐藏效果不佳,可能会带来相当多负面的效果;能见度设定得太紧,造成效率变差、扩充程度变差。

【继承】

被继承的对象称为基类(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++中,你可以继承许多类,不需要再定义这些方法。

[关于C++多重继承的底层实现机理,请看这里]

设计继承时,必须先考虑接口是否共享,再考虑代码是否共享,再考虑分类。但是一般的程序员,反倒会先考虑分类和代码复用,而忽略了"接口复用"是其中最重要的事。

【多态Polymorphism】

多态是允许你将父对象设置成为和一个或更多的他的子对象相等的技术,赋值之后,父对象就可以根据当前赋值给它的子对象的特性以不同的方式运作。简单的说,就是一句话:允许将子类类型的指针赋值给父类类型的指针

多态让人觉得对象可以以多种面貌出现。当一个对象具有不同的型态,就有可能会引发多态机制。

一个对象为何为有不同的型态?这是因为继承而来,对象可以扮演所有祖先类的角色。例如当某对象的类是派生类,当此对象被“转型”成基类之后,此对象就具有两种不同的类,“实际类”是派生类,“形式类”是基类,此时调用此对象的方法m,会执行到的是基类定义的方法m?还是派生类定义(修改)的方法m?

答案是实际类的方法,也就是派生类定义的方法m。所以所谓的多态就是:不管形式类是什么,一定会执行到实际类的方法。

你可能会觉得疑惑,为何当初要将对象转型为祖先类,导致“形式类”(宣告类)和“实际类”(定义类)不一样?

因为我们想用一个东西(基类)来表示多个东西(派生类)。

更重要的是,良好的设计是面向接口编程。我们在基类定义了接口,在派生类实现,不同的派生类可以有不同的实现。我们可以让接口的使用者不变(使用对象的“形式类”),而改变其“实际类”,改变类的行为。这样可以使程序易于修改和扩展。[比如,框架。]

类的方法,可以分成虚拟(Virtual)与非虚拟两种。只有虚拟方法才能搭配多态机制使用。如果是非虚拟方法,则会执行到形式类(而非实际类)的方法,因为多态没有发挥作用。

关于虚拟,每个语言有不同的作法。 Java强调动态,所以默认是虚拟;C++注重效率,所以默认不是虚拟。

切记「接口复用」比「代码复用」更重要。这是因为多态的缘故,多态才是OO的终极目的

[话外:关于重载与覆盖]

覆盖,是指子类重新定义父类的虚函数的做法。(我不喜欢“覆盖”这个名字,容易把人搞晕)

重载,是指允许存在多个同名函数,而这些函数的参数表不同

重载的实现是:编译器根据函数不同的参数表,对同名函数的名称做修饰,然后这些同名函数就成了不同的函数(至少对于编译器来说是这样的)。对于这两个函数的调用,在编译器间就已经确定了,是静态的(记住:是静态)。也就是说,它们的地址在编译期就绑定了(早绑定),因此,重载和多态无关!

真正和多态相关的是“覆盖”。当子类重新定义了父类的虚函数后,父类指针根据赋给它的不同的子类指针,动态(记住:是动态!)的调用属于子类的该函数,这样的函数调用在编译期间是无法确定的(调用的子类的虚函数的地址无法给出)。因此,这样的函数地址是在运行期绑定的(晚邦定)。结论就是:重载只是一种语言特性,与多态无关,与面向对象也无关!引用一句Bruce Eckel的话:“不要犯傻,如果它不是晚邦定,它就不是多态。

[注:本段改编自(http://www.cnblogs.com/clongge/archive/2008/07/09/1239076.html)]

【框架】

为了方便软件的开发,许多软件厂商都会提供应用框架(ApplicationFramework),现今流行的框架相当多,例如:.NET Framework等。有了框架,我们终于可以享受到OO的好处,重复利用别人写好的代码,不用一切自己重头写。

框架厂商先将一大部分的程序先写好,程序员只需要“利用继承来做修改”,就能套用整个框架,为了要让你修改的部分能够确实被执行到(而不是执行到框架本身的方法),所以这些允许修改的方法都是定义成虚拟的。

因为程序员“利用继承来做修改”所以产生了派生类和重新定义的方法。框架比这个派生类更早被定义,当然不认识这个派生类,所以框架内都是以此派生类的祖先类为“形式上”的处理对象(处理接口)。当此派生类对象被传入框架中,就会被自动转型成为祖先类,因此产生“形式类”和“实际类”的差异。正因为这样的类差异,加上次类有重新定义方法,所以多态机制出现了。

【乱谈面向对象】

面向过程编程和面向对象编程,是两种不同的世界观。从面向过程来看,解决一个问题是分多个步骤的流程。而从面向对象来看,解决一个问题靠的是相关类的协作。

面向对象是对现实世界直观的模拟,面向对象程序中核心的问题是,类与类之间的关系。

不同的事物有不同的属性,且有不同的能力(类的方法)。对象与对象之间传递消息,根据收到的信息,做出不同的反应。由此协作来解决问题。

类是有相同属性的对象的一种抽象,而基类之于派生类,是在一些派生类中提取共性而成的基类。是一种更高层次的抽象。

抽象,是人类理解改造这个世界的方法。人类总是试图用最简单的理论解释更多的现象。

各个学科和领域也都有相应的理论(比如热力学方程、电磁学方程),它们是对某些范围的事物的一种抽象。牛顿的万有引力,是对所有事物的一种抽象,他提取了万物的共性。这是最高层次的抽象。

(现在比较热门的“弦论”有望成为包罗所有理论的大一统理论,它也许是人类对宇宙的终极抽象吧。^_^)

[:以上内容有参考网上的,有参考书中的,为平日所搜集,今日之整理,外加自己的一些乱谈。作为一个初级菜鸟,有些地方的理解可能是有问题的,希望大家不吝指正。]

也谈面向对象

时间: 2024-07-31 10:35:16

也谈面向对象的相关文章

Java基础——再谈面向对象

去年的这个时候,心血来潮写了篇<简述面向对象技术>,先在看来不由的会想:这都是写的什么跟什么啊?(ps:虽然现在写的博客依然不咋地)但是,Java的学习中又一次不得不再一次面向对象,所以,奉上一篇<再谈面向对象>,做为新年的一盘开胃菜. 面向对象是相对于面向过程而言,是一种思想. 区别于面向过程: 面向过程是以函数为基础,完成各种操作,强调的是过程,而面向对象是以对象为基础,强调的是对象. 比如说把大象装进冰箱分为几步,宋丹丹是这样说的:三步呗, 第一步:打开冰箱门, 第二步:把大

在Objective-C中浅谈面向对象

接触面向对象也有一段时间了,当时是通过C++学习的OOP,后来又接触到了PHP和Java.每种OOP的语言在面向对象上或多或少都会有不同的地方,现在在学习OC的面向对象部分,又感觉到OC面向对象的特点.写篇博文总结一下OC中的面向对象.刚接触OC,用OC中的便利初始化方法和便利构造器有点蹩脚,不过还可以在接受的范围之内,以下的东西可能会对面向对象的特征:抽象,封装,继承等总结的较少一些,主要总结了OC中面向对象的特点.在用到便利构造器的时候,如果之前学习过设计模式的话会好理解一些. 在下面的代码

Mysql数据库大量删除操作及谈面向对象中的封装继承和多态原理(图)

Mysql数据库大量删除操作及谈面向对象中的封装继承和多态原理(图)最近进行数据库操作,遇到一个问题,就是大量删除一个数据表中的数据后,由于设定了id是自增的,导致再插入时,默认生成的id会很大,这个时候想要再次插入新的数据,应该怎么办呢?1.明确目前最后一个id的大小select id from tags order by id DESC limit 0,1; 假设返回的是9,则设置数据表从10开始自增 2.修改这个数据表从10开始自增alter table tags auto_increme

再谈面向对象多态及C++实践

多态特性: 提起面向对象,很自然地想到三大特性:封装.继承.多态.他们的目录分别是: 1. 封装,使代码模块化封装内部结构和状态. 2. 继承,用于扩展原有代码. 3. 多态,方便接口重用,通过同一接口和传入的对象调用适用于不同对象的实现.多态在运行时绑定函数,而非多态则在编译期就已确定了函数的调用地址. 从架构设计的层面来看多态有什么好处呢?将源码和运行的依赖进行反转. 通常在系统中一个函数调用另一个函数,不论是运行时还是源码中都以是相同方向进行依赖,即调用模块依赖于被调用模块. 而多态则使得

day10 浅谈面向对象编程

面向对象编程:第一步找名词,名词是问题域中的. 第二步概括名词设计成类.某些名词可以浓缩包含到其它名词中,成为其属性. 第三步找动词,动词也是问题域中的.   第四步概括动词设计成方法.动作的产生往往是对象身上发生的,根据动词动作的产生归纳到所属对象. 第五步根据需求确立方法的参数和返回值.如果在调用的地方,不需要值的返回或者返回的值后面根本用不到,则使用void.反之,如果方法调用的地方需要返回值或者后面其它地方需要用到该方法的结果,则设置相应的返回类型.如果方法中要使用其它地方传进来的值数据

虚拟世界(代码)--浅谈面向对象编程感触

不知不觉已经来到北京一年了,接触编程也一年了,或许我已经不再是那个连HelloWord都要写了N遍都不理解的低级菜鸟了(因为现在是中级了!!~~依旧是菜鸟) 在之前从来没有接触过编程,也不知道什么面向对象编程,面向过程编程.我的代码人生就是从这面向对象编程开始的. 面向对象编程,接触了C#和Java两种编程语言,感触良多(说实话,我现在已经快记不得老师讲过的有哪些重点了).我根据自己所了解的谈谈这段时间的收获. 一.什么是面向对象编程? 最初,老师告诉我们C#和Java都是面向对象编程的语言.那

Java系列1 -- 浅谈面向对象

也许每一个计算机专业的人,在大学学习java的时候,老师开始时都会说这么一句话,"Java是一门面向对象的语言".那么面向对象到底是什么,他有什么好处,或者他比其他流行的语言C/C++他强在哪里呢?“面向对象”在我们实际编程中有什么作用呢? 在开始实际的java之旅前,我们再次老生长谈下.我就按我自己的理解结合我这两年左右的编程经验来简单的说一说.其实在java之前我是学习C和C++的,但是并没有实际拿C或者C++写过什么项目,所以,一开始我就是直接用java的,在第一年的时候,我对面

浅谈面向对象

面向对象:顾名思义就是站在对象的角度思考问题,我们把多个功能合理的放到不同对象里,强调的是具备某些功能的对象. 具备某种功能的实体,称为对象, 面向对象很符合我们常规的思维方式,稳定性好,可重用性强,易于开发大型软件产品,有良好的可维护性和可拓展性. 面向对象的三大特征:封装,继承,多态 封装:影响对象的属性和实现细节,仅提供一个对外的接口,保证数据的安全性和完整性: 继承:实现一个类的时候可以在另外一个类的基础上实现,当子类继承父类后,子类是一种特殊的父类,能直接或间接获得父类里的成员,易于程

【Java基础】--再谈面向对象

面向对象总结 V1.0     2014.9.14 面向对象总结V2.0   2015.8.14 对照之前的J2SE总结,发现现在的似乎更加的注重联系,開始能把细节的东西都编制到知识网络里面,导图的图片真的非常赏心悦目.