在java以及其余的面向对象设计模式中,类与类之间主要有6种关系,他们分别是:依赖、关联、聚合、组合、继承、实现。他们的耦合度依次加强。java
1. 依赖(Dependence) 编程
依赖关系的定义为:对于两个相对独立的对象,当一个对象负责构造另外一个对象的实例,或者依赖另外一个对象的服务时,这两个对象之间主要体现为依赖关系。定义比较晦涩难懂,但在java中的表现仍是比较直观的:类A当中使用了类B,其中类B是做为类A的方法参数、方法中的局部变量、或者静态方法调用。类上面的图例中:People类依赖于Book类和Food类,Book类和Food类是做为类中方法的参数形式出如今People类中的。设计模式
代码样例:app
[java] view plaincopy编程语言
2.关联(Association)、
、
学习
单向关联:spa
双向关联:.net
对于两个相对独立的对象,当一个对象的实例与另外一个对象的一些特定实例存在固定的对应关系时,这两个对象之间为关联关系。关联关系分为单向关联和双向关联。在java中,单向关联表现为:类A当中使用了类B,其中类B是做为类A的成员变量。双向关联表现为:类A当中使用了类B做为成员变量;同时类B中也使用了类A做为成员变量。设计
代码样例:面向对象设计模式
[java] view plaincopy
3.聚合(Aggregation)
聚合关系是关联关系的一种,耦合度强于关联,他们的代码表现是相同的,仅仅是在语义上有所区别:关联关系的对象间是相互独立的,而聚合关系的对象之间存在着包容关系,他们之间是“总体-个体”的相互关系。
代码样例:
[java] view plaincopy
4.组合(Composition)
相比于聚合,组合是一种耦合度更强的关联关系。存在组合关系的类表示“总体-部分”的关联关系,“总体”负责“部分”的生命周期,他们之间是共生共死的;而且“部分”单独存在时没有任何意义。在下图的例子中,People与Soul、Body之间是组合关系,当人的生命周期开始时,必须同时有灵魂和肉体;当人的生命周期结束时,灵魂肉体随之消亡;不管是灵魂仍是肉体,都不能单独存在,他们必须做为人的组成部分存在。
[java] view plaincopy
5.继承(Generalization)
继承表示类与类(或者接口与接口)之间的父子关系。在java中,用关键字extends表示继承关系。UML图例中,继承关系用实线+空心箭头表示,箭头指向父类。
6.实现(Implementation)
表示一个类实现一个或多个接口的方法。接口定义好操做的集合,由实现类去完成接口的具体操做。在java中使用implements表示。UML图例中,实现关系用虚线+空心箭头表示,箭头指向接口。
在java中继承使用extends关键字,实现使用implements关键字,很直观。
在学习面向对象设计对象关系时,依赖、关联、聚合和组合这四种关系之间区别比较容易混淆。特别是后三种,仅仅是在语义上有所区别,所谓语义就是指上下文环境、特定情景等。他们在编程语言中的体现倒是基本相同的,可是基本相同并不等于彻底相同。
下面就来详细的论述一下在java中如何准确的体现依赖、关联、聚合和组合。
首先看一看书上对这四种关系的定义:
——摘自《Java面向对象编程》,做者:孙卫琴
以上关系的耦合度依次加强(关于耦合度的概念将在之后具体讨论,这里能够暂时理解为当一个类发生变动时,对其余类形成的影响程度,影响越小则耦合度越弱,影响越大耦合度越强)。由定义咱们已经知道,依赖关系其实是一种比较弱的关联,聚合是一种比较强的关联,而组合则是一种更强的关联,因此笼统的来区分的话,实际上这四种关系、都是关联关系。
依赖关系比较好区分,它是耦合度最弱的一种,在java中表现为局域变量、方法的形参,或者对静态方法的调用,以下面的例子:Driver类依赖于Car类,Driver的三个方法分别演示了依赖关系的三种不一样形式。
[java] view plaincopy
关联关系在java中通常使用成员变量来实现,有时也用方法形参的形式实现。依然使用Driver和Car的例子,使用方法参数形式能够表示依赖关系,也能够表示关联关系,毕竟咱们没法在程序中太准确的表达语义。在本例中,使用成员变量表达这个意思:车是我本身的车,我“拥有”这个车。使用方法参数表达:车不是个人,我只是个司机,别人给我什么车我就开什么车,我使用这个车。
[java] view plaincopy
聚合关系是是一种比较强的关联关系,java中通常使用成员变量形式实现。对象之间存在着总体与部分的关系。例如上例中
[java] view plaincopy
假如给上面代码赋予以下语义:车是一辆私家车,是司机财产的一部分。则相同的代码即表示聚合关系了。聚合关系通常使用setter方法给成员变量赋值。
假如赋予以下语义:车是司机的必须有的财产,要想成为一个司机必需要先有辆车,车要是没了,司机也不想活了。并且司机要是不干司机了,这个车就砸了,别人谁也别想用。那就表示组合关系了。通常来讲,为了表示组合关系,经常会使用构造方法来达到初始化的目的,例如上例中,加上一个以Car为参数的构造方法
[java] view plaincopy
因此,关联、聚合、组合只能配合语义,结合上下文才可以判断出来,而只给出一段代码让咱们判断是关联,聚合,仍是组合关系,则是没法判断的。