-
组合、继承和代理三者的定义:
- 组合:在新类中new 另外一个类的对象,以添加该对象的特性。
- 继承:从基类继承得到子类,获得基类的特性。
- 代理:在代理类中创建某功能的类,调用类的一些方法以获得该类的部分特性。
- 使用场合:
- 组合:各部件之间没什么关系,只需要组合即可。like组装电脑,需要new CPU(),new RAM(),new Disk()……
- 继承:子类需要具有父类的功能,各子类之间有所差异。like Shape类作为基类,子类有Rectangle,CirCle,Triangle……代码不写了,大家都经常用。
- 代理:飞机控制类,我不想暴露太多飞机控制的功能,只需部分前进左右转的控制(而不需要暴露发射导弹功能)。通过在代理类中new一个飞机控制对象,然后在方法中添加飞机控制类的各个需要暴露的功能。
-
一:继承
继承按现有类创建一个对象,不改变现有类,采用现有类的形式向其中添加新代码;(编译器强制你去初始化基类,是is_a 的关系,比如说Student继承Person,则说明Student is a Person。
继承的优点是子类可以重写父类的方法来方便地实现对父类的扩展。
继承的缺点有以下几点:
①:父类的内部细节对子类是可见的。
②:子类从父类继承的方法在编译时就确定下来了,所以无法在运行期间改变从父类继承的方法的行为。
③:如果对父类的方法做了修改的话(比如增加了一个参数),则子类的方法必须做出相应的修改。所以说子类与父类是一种高耦合,违背了面向对象思想。
二:组合
组合是在新类中使用现有类的对象,也就是设计类的时候把要组合的类的对象加入到该类中作为自己的成员变量。组合是has_a 的关系。
组合的优点:
①:当前对象只能通过所包含的那个对象去调用其方法,所以所包含的对象的内部细节对当前对象时不可见的。
②:当前对象与包含的对象是一个低耦合关系,如果修改包含对象的类中代码不需要修改当前对象类的代码。
③:当前对象可以在运行时动态的绑定所包含的对象。可以通过set方法给所包含对象赋值。
组合的缺点:①:容易产生过多的对象。②:为了能组合多个对象,必须仔细对接口进行定义。
由此可见,组合比继承更具灵活性和稳定性,所以在设计的时候优先使用组合。
只有当下列条件满足时才考虑使用继承:
子类是一种特殊的类型,而不只是父类的一个角色
子类的实例不需要变成另一个类的对象
子类扩展,而不是覆盖或者使父类的功能失效InheritTest.java 使用继承方式实现目标
- //InheritTest.java 使用继承方式实现目标
- class Animal{
- private void beat(){
- System.out.println("心脏跳动...");
- }
- public void breath(){
- beat();
- System.out.println("吸一口气,呼一口气,呼吸中...");
- }
- }
- //继承Animal,直接复用父类的breath()方法
- class Bird extends Animal{
- //创建子类独有的方法fly()
- public void fly(){
- System.out.println("我是鸟,我在天空中自由的飞翔...");
- }
- }
- //继承Animal,直接复用父类的breath()方法
- class Wolf extends Animal{
- //创建子类独有的方法run()
- public void run(){
- System.out.println("我是狼,我在草原上快速奔跑...");
- }
- }
- public class InheritTest{
- public static void main(String[] args){
- //创建继承自Animal的Bird对象新实例b
- Bird b=new Bird();
- //新对象实例b可以breath()
- b.breath();
- //新对象实例b可以fly()
- b.fly();
- Wolf w=new Wolf();
- w.breath();
- w.run();
- /*
- ---------- 运行Java程序 ----------
- 心脏跳动...
- 吸一口气,呼一口气,呼吸中...
- 我是鸟,我在天空中自由的飞翔...
- 心脏跳动...
- 吸一口气,呼一口气,呼吸中...
- 我是狼,我在草原上快速奔跑...
- 输出完毕 (耗时 0 秒) - 正常终止
- */
- }
- }
CompositeTest.java 使用组合方式实现目标
- //CompositeTest.java 使用组合方式实现目标
- class Animal{
- private void beat(){
- System.out.println("心脏跳动...");
- }
- public void breath(){
- beat();
- System.out.println("吸一口气,呼一口气,呼吸中...");
- }
- }
- class Bird{
- //定义一个Animal成员变量,以供组合之用
- private Animal a;
- //使用构造函数初始化成员变量
- public Bird(Animal a){
- this.a=a;
- }
- //通过调用成员变量的固有方法(a.breath())使新类具有相同的功能(breath())
- public void breath(){
- a.breath();
- }
- //为新类增加新的方法
- public void fly(){
- System.out.println("我是鸟,我在天空中自由的飞翔...");
- }
- }
- class Wolf{
- private Animal a;
- public Wolf(Animal a){
- this.a=a;
- }
- public void breath(){
- a.breath();
- }
- public void run(){
- System.out.println("我是狼,我在草原上快速奔跑...");
- }
- }
- public class CompositeTest{
- public static void main(String[] args){
- //显式创建被组合的对象实例a1
- Animal a1=new Animal();
- //以a1为基础组合出新对象实例b
- Bird b=new Bird(a1);
- //新对象实例b可以breath()
- b.breath();
- //新对象实例b可以fly()
- b.fly();
- Animal a2=new Animal();
- Wolf w=new Wolf(a2);
- w.breath();
- w.run();
- /*
- ---------- 运行Java程序 ----------
- 心脏跳动...
- 吸一口气,呼一口气,呼吸中...
- 我是鸟,我在天空中自由的飞翔...
- 心脏跳动...
- 吸一口气,呼一口气,呼吸中...
- 我是狼,我在草原上快速奔跑...
- 输出完毕 (耗时 0 秒) - 正常终止
- */
- }
- }
组合和继承都允许在新的类中放置子对象,组合是显式地这样做,而继承则是隐式地这样做。
继承是"is-a"关系
组合是"has-a"关系
1)组合(has-a)关系可以显式地获得被包含类(继承中称为父类)的对象,而继承(is-a)则是隐式地获得父类的对象,被包含类和父类对应,而组合外部类和子类对应。
2)组合关系在运行期决定,而继承关系在编译期就已经决定了。
3)组合是在组合类和被包含类之间的一种松耦合关系,而继承则是父类和子类之间的一种紧耦合关系。
4)当选择使用组合关系时,在组合类中包含了外部类的对象,组合类可以调用外部类必须的方法,而使用继承关系时,父类的所有方法和变量都被子类无条件继承,子类不能选择。
5)最重要的一点,使用继承关系时,可以实现类型的回溯,即用父类变量引用子类对象,这样便可以实现多态,而组合没有这个特性。
6)还有一点需要注意,如果你确定复用另外一个类的方法永远不需要改变时,应该使用组合,因为组合只是简单地复用被包含类的接口,而继承除了复用父类的接口外,它甚至还可以覆盖这些接口,修改父类接口的默认实现,这个特性是组合所不具有的。
7)从逻辑上看,组合最主要地体现的是一种整体和部分的思想,例如在电脑类是由内存类,CPU类,硬盘类等等组成的,而继承则体现的是一种可以回溯的父子关系,子类也是父类的一个对象。8)这两者的区别主要体现在类的抽象阶段,在分析类之间的关系时就应该确定是采用组合还是采用继承。
9)引用网友的一句很经典的话应该更能让大家分清继承和组合的区别:组合可以被说成“我请了个老头在我家里干活” ,继承则是“我父亲在家里帮我干活"。
10)组合技术通常是用于在新类中使用现有类的功能而非它的接口的情形。
三、代理
代理是继承和组合的中庸之道。将一个成员对象置于要构造的类中(就像组合),与此同时在新类中暴露了该类成员所有的方法(就像继承)。
(组合+继承)java中没有直接的支持代理,(将基类对象作为代理类的成员,而代理类有对应于基类的所有方法,各方法内使用基类对象成员调用基类的方法。)
代理的实例一:
- public class test {
- public static void main(String[] args) {
- Demo2 d2=new Demo2();
- d2.print();
- //组合
- Demo1 d1=new Demo1();
- d1.method();
- //代理
- Demo3 d3=new Demo3();
- d3.stop();
- }
- }
- class Demo{
- Demo(){System.out.println("hello world!");}
- public static void stop(){System.out.println("Hello stop!");}
- }
- class Demo1{ //组合机制
- public static void method(){
- Demo d=new Demo();//在新类中创建现有类的对象
- }
- }
- class Demo2 extends Demo{ //继承机制,按照现有类来创建新类,无需改变现有类的形式,采用现有类的形式并且在其中添加新代码
- public static void print(){
- System.out.println("YES!");
- }
- }
- class Demo3{
- private Demo d=new Demo();
- public void stop(){ //代理,既有继承功能 又有组合功能
- System.out.println("Hello 继承!");
- d.stop(); }}
输出结果:
hello world!
YES!
hello world!
hello world!
hello 继承!
hello stop!
图文总结: