1.继承
1.1继承的定义
继承(inheritance)机制:是面向对象程序设计使代码可以复用的最重要的手段,它允许程序员在保持原有类特性的基础上进行扩展,增加新功能,这样产生新的类,称派生类。继承呈现了面向对象程序设计的层次结构, 体现了 由简单到复杂的认知过程。继承主要解决的问题是:共性的抽取,实现代码复用。
在写代码的时候会有大量的代码出现多次,使得代码出现冗余,而面向对象思想中提出了继承的概念,专门用来进行共性抽取,实现代码复用,减轻代码的冗余。
1.2继承的语法
在Java中如果要表示类之间的继承关系,需要借助extends关键字,具体如下:
修饰符 class 子类 extends 父类 {
// ...
}
继承代码演示:
class School { public String name; public String classRoom; public void sleep() { System.out.println(name+"休息"); } } class Teacher extends School { public void teach() { System.out.println(name+"教学"); } } class Student extends School { public void learn() { System.out.println(name+"学习"); } } public class Test_11_12 { public static void main(String[] args) { Teacher t1=new Teacher(); t1.name="大华"; t1.sleep(); t1.teach(); Student s1=new Student(); s1.name="小明"; s1.sleep(); s1.learn(); } }
注意:
- 1.子类会将父类中的成员变量或者成员方法继承到子类中
- 2.子类继承父类之后,必须要新添加自己特有的成员,体现出与基类的不同,否则就没有必要继承了
1.3父类成员访问
1.3.1 子类中访问父类的成员变量
1. 子类和父类不存在同名成员变量
public class Base { int a; int b; } public class Derived extends Base{ int c; public void method(){ a = 10; // 访问从父类中继承下来的a b = 20; // 访问从父类中继承下来的b c = 30; // 访问子类自己的c } }
2. 子类和父类成员变量同名
public class Base { int a; int b; int c; } / public class Derived extends Base{ int a; // 与父类中成员a同名,且类型相同 char b; // 与父类中成员b同名,但类型不同 public void method(){ a = 100; // 访问父类继承的a,还是子类自己新增的a? b = 101; // 访问父类继承的b,还是子类自己新增的b? c = 102; // 子类没有c,访问的肯定是从父类继承下来的c // d = 103; // 编译失败,因为父类和子类都没有定义成员变量b } }
在子类方法中 或者 通过子类对象访问成员时 注意:
- 如果访问的成员变量子类中有,优先访问自己的成员变量
- 如果访问的成员变量子类中无,则访问父类继承下来的,如果父类也没有定义,则编译报错。
- 如果访问的成员变量与父类中成员变量同名,则优先访问自己的(就近原则)。。
1.3.2 子类中访问父类的成员方法
1. 成员方法名字不同
public class Base { public void methodA(){ System.out.println("Base中的methodA()"); } } public class Derived extends Base{ public void methodB(){ System.out.println("Derived中的methodB()方法"); } public void methodC(){ methodB(); // 访问子类自己的methodB() methodA(); // 访问父类继承的methodA() // methodD(); // 编译失败,在整个继承体系中没有发现方法methodD() } }
总结:在访问成员方法时,先在子类中寻找有无这个方法,有就调运自己的,无就再去父类中寻找有无这个方法,有调用父类的,无则报错。
2. 成员方法名字相同
public class Base { public void methodA(){ System.out.println("Base中的methodA()"); } public void methodB(){ System.out.println("Base中的methodB()"); } } public class Derived extends Base{ public void methodA(int a) { System.out.println("Derived中的method(int)方法"); } public void methodB(){ System.out.println("Derived中的methodB()方法"); } public void methodC(){ methodA(); // 没有传参,访问父类中的methodA() methodA(20); // 传递int参数,访问子类中的methodA(int) methodB(); // 直接访问,则永远访问到的都是子类中的methodB(),基类的无法访问到 } }
总结:通过派生类对象访问父类与子类同名方法时,如果父类和子类同名方法的参数列表不同(重载),根据调用方法适传递的参数选择合适的方法访问,如果没有则报错。
1.4super关键字
由于设计不好,或者因场景需要,子类和父类中可能会存在相同名称的成员,如果要在子类方法中访问父类同名成员时,直接访问是无法做到的,Java提供了super关键字,该关键字主要作用:在子类方法中访问父类的成员。
class School { int age=28; public String name; public String classRoom; public void sleep() { System.out.println(name+"休息"); } } class Teacher extends School { int age=18; public void test() { System.out.println(age);//打印子类中的age System.out.println(super.age);//打印父类中的age } public void teach() { System.out.println(name+"教学"); } }
注意事项
- 1.只能在非静态方法中使用。
- 2.在子类方法中,访问父类的成员变量和方法。
1.5子类构造方法
在继承关系中,当我们构造子类时,要先帮父类构造。构造的方式是在子类中调用父类的构造方法。
public class Base { public Base(){ System.out.println("Base()"); } } public class Derived extends Base{ public Derived(){ // super(); // 注意子类构造方法中默认会调用基类的无参构造方法:super(), // 用户没有写时,编译器会自动添加,而且super()必须是子类构造方法中第一条语句, // 并且只能出现一次 System.out.println("Derived()"); } } public class Test { public static void main(String[] args) { Derived d = new Derived(); } }
运行结果:
通过运行结果可以知道:在构造子类对象时候 ,先要调用基类的构造方法,将从基类继承下来的成员构造完整 ,然后再调用子类自己的构造方法,将子类自己新增加的成员初始化完整 。
注意:
1.若父类显式定义无参或者默认的构造方法,在子类构造方法第一行默认有隐含的super()调用,即调用基类构造方法。
2.如果父类构造方法是带有参数的,此时需要用户为子类显式定义构造方法,并在子类构造方法中选择合适的父类构造方法调用(重载),否则编译失败。
3.在子类构造方法中,super(...)调用父类构造时,必须是子类构造函数中第一条语句。
4.super(...)只能在子类构造方法中出现一次,并且不能和this同时出现。
1.6super和this对比
相同点:
1.都是Java中的关键字
2.只能在类的非静态方法中使用,用来访问非静态成员方法和字段
3.在构造方法中调用时,必须是构造方法中的第一条语句,并且不能同时存在
不同点:
1. this是当前对象的引用,当前对象即调用实例方法的对象,super相当于是子类对象中从父类继承下来部分成员的引用。
2. 在非静态成员方法中,this用来访问本类的方法和属性,super用来访问父类继承下来的方法和属性。
3. 在构造方法中:this(...)用于调用本类构造方法,super(...)用于调用父类构造方法,两种调用不能同时在构造方法中出现。
4. 构造方法中一定会存在super(...)的调用,用户没有写编译器也会增加,但是this(...)用户不写则没有。
1.7继承方式
在现实生活中,事物之间的关系是非常复杂,灵活多样,比如:
但在Java中只支持以下几种继承方式:
单继承:
public class A { //... } public class B extends A { //... }
多层继承:
public class A { //... } public class B extends A { //... } public class C extends B { //... }
不同类继承:
public class A { //... } public class B extends A { //... } public class C extends A{ //... }
多继承(Java不支持):
public class A { //... } public class B { //... } public class C extends A,B { //... }//不支持
注意:Java中不支持多继承。