【JavaSE】一起学继承
前言:
作者简介:爱吃大白菜1132
人生格言:纸上得来终觉浅,绝知此事要躬行
如果文章知识点有错误的地方不吝赐教,和大家一起学习,一起进步!
如果觉得博主文章还不错的话,希望三连支持!
目录
1.继承
1.1继承的定义
1.2继承的语法
1.3父类成员访问
1.3.1 子类中访问父类的成员变量
1.3.2 子类中访问父类的成员方法
1.4super关键字
1.5子类构造方法
1.6super和this对比
1.7继承方式
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.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.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();
}
}
运行结果:
通过运行结果可以知道:在构造子类对象时候 ,先要调用基类的构造方法,将从基类继承下来的成员构造完整 ,然后再调用子类自己的构造方法,将子类自己新增加的成员初始化完整 。
注意:
- 若父类显式定义无参或者默认的构造方法,在子类构造方法第一行默认有隐含的super()调用,即调用基类构造方法。
- 如果父类构造方法是带有参数的,此时需要用户为子类显式定义构造方法,并在子类构造方法中选择合适的父类构造方法调用(重载),否则编译失败。
- 在子类构造方法中,super(...)调用父类构造时,必须是子类构造函数中第一条语句。
- super(...)只能在子类构造方法中出现一次,并且不能和this同时出现。
1.6super和this对比
相同点:
- 都是Java中的关键字
- 只能在类的非静态方法中使用,用来访问非静态成员方法和字段
- 在构造方法中调用时,必须是构造方法中的第一条语句,并且不能同时存在
不同点:
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中不支持多继承。