面向对象的三个特征及其优缺点

来源:互联网 发布:淘宝买快排配件有事吗 编辑:程序博客网 时间:2024/05/30 04:50

面向对象(OOP)三个基本特征 - 樱空释 - 樱空释

1、封装
      把客观的事物封装成抽象的类;类可以将自身的数据和方法给可信的类或者对象操作,对不可信的进行隐藏信息。
2、继承
      可以在现有类的基础上无需重写,扩展类的功能.
      通过继承创建的新类称之为"子类"、"派生类".    被继承的类称之为"基类"、"父类"、"超类';
     
     继承实现方式有三种:
     1)、 实现继承是使用基类的属性和方法而无需额外编码;
     2)、 接口继承是仅使用属性和方法的名称、但是子类必须提供实现;
     3)、 可视继承是子窗体(类)使用基窗体(类)的外观和实现代码。
    使用继承的过程中两个类的关系是"属于"的关系,
    抽象类仅定义将由子类创建的一般属性和方法,创建抽象类时,使用关键字 Interface.
3、多态
     允许将子类类型的指针赋值给父类类型的指针。
    
     实现多态的方式:
    1)、覆盖指子类重新定义父类的虚函数的方法。
    2)、重载指允许存在多个同名函数,而这些函数的参数表不同(或许参数个数不同,或许参数类型不同,或许两者都不同)。
           重载的概念并不属于“面向对象编程”,重载的实现是:编译器根据函数不同的参数表,对同名函数的名称进行修饰,然后这些同名函数就成了不同的函数。
    如,有两个同名函数:function func(p:integer):integer;和function func(p:string):integer;。那么编译器做过修饰后的函数名称可能是这样的:int_func、str_func。对于这两个函数的调用,在编译器间就已经确定了,是静态的。也就是说,它们的地址在编译期就绑定了,因此,重载和多态无关!真正和多态相关的是“覆盖”。当子类重新定义了父类的虚函数后,父类指针根据赋给它的不同的子类指针,动态(记住:是动态!)的调用属于子类的该函数,这样的函数调用在编译期间是无法确定的(调用的子类的虚函数的地址无法给出)。因此,这样的函数地址是在运行期绑定的。结论就是:重载只是一种语言特性,与多态无关,与面向对象也无关!
 

概念讲解

泛化(Generalization)

o_Generalization.gif

图表 1 泛化

在上图中,空心的三角表示继承关系(类继承),在UML的术语中,这种关系被称为泛化(Generalization)。Person(人)是基类,Teacher(教师)、Student(学生)、Guest(来宾)是子类。

若在逻辑上B是A的“一种”,并且A的所有功能和属性对B而言都有意义,则允许B继承A的功能和属性。

例如,教师是人,Teacher 是Person的“一种”(a kind of )。那么类Teacher可以从类Person派生(继承)。

如果A是基类,B是A的派生类,那么B将继承A的数据和函数。

如果类A和类B毫不相关,不可以为了使B的功能更多些而让B继承A的功能和属性。

若在逻辑上B是A的“一种”(a kind of ),则允许B继承A的功能和属性。

 

聚合(组合)

o_aggregationBase.gif

图表 2 组合

若在逻辑上A是B的“一部分”(a part of),则不允许B从A派生,而是要用A和其它东西组合出B。

例如,眼(Eye)、鼻(Nose)、口(Mouth)、耳(Ear)是头(Head)的一部分,所以类Head应该由类Eye、Nose、Mouth、Ear组合而成,不是派生(继承)而成。

 

聚合的类型分为无、共享(聚合)、复合(组合)三类。

 

聚合(aggregation)

 

o_aggregation.gif

图表 3 共享

上面图中,有一个菱形(空心)表示聚合(aggregation)(聚合类型为共享),聚合的意义表示has-a关系。聚合是一种相对松散的关系,聚合类B不需要对被聚合的类A负责。

 

组合(composition)

o_composition.gif

图表 4 复合

这幅图与上面的唯一区别是菱形为实心的,它代表了一种更为坚固的关系——组合(composition)(聚合类型为复合)。组合表示的关系也是has-a,不过在这里,A的生命期受B控制。即A会随着B的创建而创建,随B的消亡而消亡。

 

依赖(Dependency)

o_Dependency.gif

图表 5 依赖

这里B与A的关系只是一种依赖(Dependency)关系,这种关系表明,如果类A被修改,那么类B会受到影响。



原创粉丝点击