欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  Java

Java-面向对象-继承、多态、组合

程序员文章站 2022-05-04 16:46:47
...

继承的特点

单继承:每个子类最多只有一个直接父类,注意是直接父类,间接父类个数不限

注意父类的概念:A-->B-->C-->D,在这里,ABC都是D的父类,C是D的直接父类,AB是D的间接父类

父类和子类是一般和特殊的关系;子类扩展了父类,子类是一种特殊的父类

Object是所有类的直接或间接父类;定义一个类时,若没有直接指定父类,则默认继承Object类

子类从父类继承了哪些要素

子类不继承父类的构造方法,但是总要调用父类的构造方法

子类继承了父类的实例变量和实例方法、类变量和类方法,当然前提是没有被private修饰

父类中被private修饰的成员,不会被子类继承

父类中用final修饰的方法,子类不能重写

方法的重写override

子类和父类拥有相同的方法签名,叫做子类重写了父类的方法

两同两小一大原则:

两同:方法名相同,形参列表相同

两小:子类方法的返回值跟父类的相等或更小;子类方法抛出的异常要跟父类抛出的相等或更小

一大:子类方法的访问权限要跟父类的相等或更大

static:子类方法要跟父类方法一致,要么是类方法,要么是实例方法,二者不能不一致

方法重写后,子类对象将无法访问父类的方法,但可以在子类方法中调用父类的方法:super.实例方法,父类名.类方法

如果父类方法是private修饰,则子类不能重写该方法;如果子类拥有一个跟父类private方法相同签名的方法,这也不是重写,这是子类新添加的方法,与父类的无关

子类方法可能会和父类的方法发生重载

父类中用final修饰的方法,子类不能重写

super关键字

在子类中,用super关键字调用父类的实例变量或实例方法

super和static不能同时用来修饰方法;就像this不能和static不能同时修饰方法一样

子类从父类继承了一个变量,又再定义了一个变量,而且同名,此时可以用super.变量名或者父类名.变量名来访问父类的这个变量

变量的查找顺序

子类方法访问了一个变量,没有显式指定调用者的情况下,按以下顺序查找:

当前方法中,是否有同名的局部变量

当前类中,是否有同名的成员变量

直接父类中,是否有

逐级网上追溯,如果最终没有找到,那么提示编译出错

super调用父类构造器

子类虽然不会继承父类的构造器,但总是会调用父类的构造器。调用同一个类的构造器用this,调用父类的用super

显式调用:super调用语句写在在构造方法的第一行,因为this也要写在第一行,因此super和this不会同时出现

间接调用:子类方法的第一行写的是this,被调用的构造器还得调用父类的构造器

隐式调用:无super无this,系统默认调用父类的无参构造

子类继承父类,总会一级一级的往上调用父类构造器,直到调用Object的,并且是找到顶部父类构造器后,开始往下一层一层执行

比如下面的代码,继承关系:A-->B-->C-->D

public class Test{  
    public static void main(String[] args) {  
        D d=new D();
    }
} 
class D extends C{    D(){
        System.out.println("D类构造器");
    }
}class C extends B{    C(){
        System.out.println("C类构造器");
    }
}class B extends A{    B(){
        System.out.println("B类构造器");
    }
}class A{    A(){
        System.out.println("A类构造器");
    }
}

输出:

A类构造器
B类构造器
C类构造器
D类构造器

如果父类没有无参构造,子类又需要调用父类的无参构造,那么不能通过编译

多态Polymorphism

简单的说,多态就是:在一个金字塔式的继承体系中,创建底部子类对象时,用顶端的父类类型指向这些底部的子类对象,通过相同类型的引用变量调用同一个方法时,会出现不同的结果

多态源于:继承+父类型的引用指向子类型的对象+方法的重写

引用变量有两个类型,一个是编译时类型,一个是运行时类型

向上转型:父类型引用指向子类型对象,向上转型由系统自动完成

向上转型的情况下:通过父类型的引用可以调用子类重写了的方法,但不能访问父类中没有子类中才有的实例变量,也就是说实例变量不具有多态性

instanceof与强制类型转换

基本类型有强制类型转换,比如(double)16;引用类型也存在强制类型转换

比如存在以下继承链:A-->B-->C-->D-->E-->F,B b=new D(),此时用B类型指向D对象,此时可以把b变量转为ACD类型,但不能转为EF类型,抛出异常ClassCastException

进行强制类型转换之前,应先用instanceof进行判断能不能转,避免抛出异常

X变量 instanceof Y类型:判断X变量能不能转为Y类型,可能出现三种情况:

true表示可以转,意味着X变量指向的对象的类型,是Y类型或者Y的子类型;

false表示不可以转,意味着X变量的类型与Y类型存在父子继承关系,不可能是子父关系,X变量指向的对象的类型与Y类型没有父子或子父关系,二者都是X变量类型的子类型;

编译不通过,表示X和Y没有父子或者子父继承关系

继承与组合

继承可以实现代码的复用,但破坏了封装;组合也可以实现代码的复用

设计父类遵循的原则:

尽量用private隐藏父类的内部数据,不让子类直接访问父类的成员变量

不要让子类随意访问、修改父类方法。父类辅助性的工具方法,用private修饰;需要让外部调用的方法,用public;不希望子类重写的方法用final修饰;希望子类重写而不希望外部访问则用protected

不要在父类构造器中调用可能被子类重写的方法,后果很严重

设计继承的原则:

子类需要增加额外的属性,而不是属性的改变

子类需要增加自己独有的功能或行为,可以增加新方法,或者重写父类方法

不要光出于代码复用的目的设计继承,而要看实际情况,父类和子类是否有"has-a"的关系

不想让一个类被继承:

final:用final修饰该类为最终类,不能被继承

private:用private修饰该类的所有构造方法,这样子类就无法调用该类的构造器,就不能继承了,另外应提供一个静态方法,用来返回该类的对象

组合也能实现代码的复用

A类复用B类的方法

A类中创建B类的对象,并用private修饰

在A类方法中调用B类对象的方法

组合还是继承:

继承:子类与父类是"is-a"关系,具体与抽象、特殊与一般的关系

组合:新类与旧类是"has-a"关系,整体与局部的关系

其他:

对于子类从父类继承了一个变量,又定义了一个同名变量的情况下,在创建子类对象时,为这两个变量都分配了内存,只是父类变量被隐藏

相关标签: Java