转载声明:文章来源:https://blog.csdn.net/qq_58710208/article/details/120410867
1. 概念
继承的基本思想是,基于已有的类创造新的类。继承已存在的类就是复用这些类的方法,而且可以增加一些新的方法和字段,使新类能够适应新的情况,继承是Java程序设计中一项核心技术,它主要解决的问题是:共性的抽取,实现代码复用。例:
上图中,dog和cat都继承Animal类,dog和cat可以称为Animal的子类或者派生类,继承之后,子类可以复用父类的方法和属性,子类在实现时只关心自己新增加的成员即可。
2. 语法
Java中表现继承的关系需要借助关键字extends:
对上述例子进行代码展示:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 | Animal.java public class Animal { String name; String gender; int age; public void eat(){ System.out.println(name+ "吃东西" ); } public void sleep(){ System.out.println(name+ "在睡觉" ); } } Dog.java public class Dog extends Animal{ void bark(){ System.out.println(name+ "旺旺~" ); } public static void main(String[] args) { Dog dog = new Dog(); System.out.println(dog.name); System.out.println(dog.gender); System.out.println(dog.age); dog.eat(); dog.sleep(); dog.bark(); } } Cat.java public class Cat extends Animal{ void mew(){ System.out.println(name+ "喵喵~" ); } public static void main(String[] args) { Cat cat = new Cat(); System.out.println(cat.name); System.out.println(cat.gender); System.out.println(cat.age); cat.eat(); cat.sleep(); cat.mew(); } } |
注意:
· 子类将继承父类的成员变量和成员方法
· 子类继承父类之后,需要添加自己特有的成员,体现出与基类的不同
3. 父类成员访问
3.1 子类访问父类的成员变量
3.1.1 子类和父类中不存在同名的成员变量
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 | public class Base { int a; int b; } public class Derived extends Base{ int c; public void method(){ a = 10; //从父类继承 b = 20; //从父类继承 c = 30; //访问自己 } } |
3.1.2 子类和父类存在相同的成员变量
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 | public class Base { int a; int b; int c; } public class Derived extends Base { char b; //与父类同名,不同类型 int c; //与父类同名,相同类型 public void method(){ a = 10; //访问父类继承 b = 20; //访问谁的? c = 30; //访问谁的? //d = 40; //编译器报错 } } |
注意:
· 如果访问的成员变量子类中有,则优先访问子类本身的
· 如果访问的成员变量子类中无,父类中有,则访问继承下来的
· 如果子类与父类中有同名的成员变量,则优先访问子类自己的,即子类将父类的同名变量隐藏
成员变量的访问遵循就近原则,自己有就优先访问自己的
3.2 子类中访问父类的成员方法
3.2.1 成员方法名字不同
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 | public class Base { public void method1(){ System.out.println( "我是父类方法" ); } } public class Derived extends Base { public void method2(){ System.out.println( "我是子类方法" ); } public void method(){ method1(); //父类方法 method2(); //子类方法 } } |
3.2.2 成员方法名字相同
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 | public class Base { public void method1(){ System.out.println( "我是父类方法" ); } public void method2(){ System.out.println( "我是父类方法" ); } } public class Derived extends Base { public void method1(int a){ System.out.println( "我是子类方法" ); } public void method2(){ System.out.println( "我是子类方法" ); } public void method(){ method1(); //父类方法 method1(10); //子类方法 method2(); //子类方法 } } |
说明:
· 通过子类访问成员方法,先看子类本身有没有,如果有访问自己的,如果没有,访问父类的
· 通过子类访问与父类同名方法时,如果子类和父类方法的参数列表不同则构成重载,根据调用方法传递的参数选择合适的方法访问
· 如果子类和父类同名方法的原型一致,则只能访问到子类的
4. super关键字
如果子类中存在与父类同名的方法成员,则通过关键字super在子类方法中访问父类方法成员
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 | public class Base { int a; int b; int c; public void method1() { System.out.println( "我是父类方法" ); } public void method2() { System.out.println( "我是父类方法" ); } } public class Derived extends Base { int a; int b; int c; public void method1(int n){ System.out.println( "我是子类方法" ); } public void method2(){ System.out.println( "我是子类方法" ); } public void method(){ a = 10; b = 20; //访问子类成员 c = 30; super .a = 10; super .b = 20; //访问父类成员 super .c = 30; method1(); //访问父类方法 method1(10); //访问子类方法 method2(); //访问子类方法 super .method2(); //访问父类方法 } } |
注意事项:只能在非静态方法中使用
5. 子类构造方法
构造哪个类的对象,就调用哪个类的构造方法,调用构造方法时先调用基类,在调用子类(即在子类中隐藏super() )
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 | public class Base { public Base(){ System.out.println( "Base()" ); } } public class Derived extends Base { public Derived(){ System.out.println( "Derived()" ); } } public class Text { public static void main(String[] args) { Derived d = new Derived(); } } 输出结果:Base() Derived() |
在子类构造方法中,并没有写任何关于基类构造的代码,但是在构造子类对象时,先执行基类的构造方法,然后执行子类的构造方法
注意:
· 若父类显示定义无参或者默认的构造方法,在子类构造方法的第一行默认有隐含的super调用,即调用基类的构造方法
· 如果父类的构造方法是带有参数的,此时编译器不会给子类生成默认的构造方法,此时需要用户在子类中显示定义构造方法,并在子类构造方法中选取合适的父类构造方法调用
· 在子类构造方法中,super(...)调用父类构造时,必须是子类构造方法中的第一条语句
· super(...)只能在子类的构造方法中出现一次,并不能和this同时出现
6. 执行顺序
无继承关系时的执行顺序:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 | public class Person { String name; String gender; int age; public Person(String name,String gender,int age){ this .name = name; this .gender = gender; this .age = age; System.out.println( "我是构造方法" ); } { System.out.println( "我是实例代码块" ); } static { System.out.println( "我是静态代码块" ); } public static void main(String[] args) { Person p1 = new Person( "xiaoHua" , "男" ,12); System.out.println( "=====================" ); Person p2 = new Person( "xiaoHong" , "女" ,15); } } |
执行结果:
说明:
静态代码块先执行,且只执行一次,在类加载阶段执行
当有对象创建时,才会执行实例代码块,实例代码块执行完后,再执行构造方法
有继承关系时的执行顺序:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 | public class Person { String name; String gender; int age; public Person(String name,String gender,int age){ this .name = name; this .gender = gender; this .age = age; System.out.println( "person的构造方法" ); } { System.out.println( "person的实例代码块" ); } static { System.out.println( "person的静态代码块" ); } } public class Student extends Person{ public Student(String name, String gender, int age) { super (name, gender, age); System.out.println( "student的构造方法" ); } { System.out.println( "student的实例代码块" ); } static { System.out.println( "student的静态代码块" ); } } public class Text { public static void main(String[] args) { Student s1 = new Student( "张三" , "男" ,35); System.out.println( "=====================" ); Student s2 = new Student( "李四" , "男" ,30); } } |
执行结果:
结论:
· 父类静态代码块优先子类静态代码块执行,都是最早执行的
· 父类实例代码块和父类构造方法紧接着执行
· 子类的实例代码块和子类构造方法在接着执行
· 第二次实例化对象时,父类和子类的静态代码块都不会在执行
7. 继承方式
Java中只支持以下几种方式:
注意:Java中不支持多支持
8. final关键字
final关键字可以用来修饰变量,成员方法以及类。
1.修饰变量或字段,表示常量(即不能修改)
2.修饰类,表示类不能继承
3.修饰方法,表示方法不能被重写
不错,慢慢看