首页 > 代码库 > Day 08Java基础学习笔记

Day 08Java基础学习笔记

继承extends(也叫扩展)

多个类中存在相同的属性和行为时,将这些内容抽取到单独一个类中,那么多个类无需再定义这些属性和行为,只要继承那个类即可。

通过extends关键字可以实现类的继承
    class 子类名 extends 父类名{}

继承的案例

创建一个Person类,定义两个功能吃饭、睡觉,再创建学生类、教师类,继承Person类,再定义两个自己独有的功能
创建Persen类
public class Person {
    public void eat(){
        System.out.println("吃饭");
    }
    public void sleep(){
        System.out.println("睡觉");
    }
}
//创建子类学生类继承Person类
public class Student extends Person {
    public void play(){
        System.out.println("打游戏");
    }
}
//创建Teacher类继承Person类
public class Teacher extends Person {
    public void teach(){
        System.out.println("教书");
    }
}
//创建测试类,对继承进行测试
public class PersonExtends {
    public static void main(String[] args) {
        Student st = new Student();
        st.eat();   //用学生类调用继承方法
        st.play();  //调用自己的方法
        Teacher t =new Teacher();
        t.sleep();
        t.teach();
    }
}

继承的好处

  • 提高了代码的复用性:多个类的相同成员可以放到一个类中

  • 提高了代码的维护性:如果功能的代码需要修改,修改一处即可,继承的类中自动都被修改了

  • 让类与类之间产生了关系,是多态的前提

Java中继承的特点

Java只支持单继承,不支持多继承

即一个类只能有一个直接父类,不可以有多个直接父类

    正确写法:class Sub extends Demo{}
    错误写法:class Sub extends Demo1,Demo2{}
Java支持多层继承
    class A{}
    class B extends A{}
    class C extends B{}
    例
    //
    public class Grandfather {
        public void show(){
            System.out.println("grandfather");
        }
    }
    public class Father extends Grandfather {
        public void method(){
            System.out.println("father");
        }
    }
    public class Son extends Father {
        public void function(){
            System.out.println("son");
        }
    }
    //测试类
    public class GrandfatherTest {
        public static void main(String[] args) {
            Son s = new Son();
            s.function();
            s.method();
            s.show();
        }
    }

Java中继承的注意事项

  1. 子类只能继承父类中所有非私有的成员(成员变量、成员方法)
  2. 子类不能父类的的构造方法,但可以同super关键字去访问父类构造方法
  3. 不要为了部分功能去继承,如果有两个类A,B,只有他们符合A是B的一种或者B是A的一种,就可以考虑使用继承

继承中成员变量的关系

案列演示:
//创建父类定义成员变量
public class Field {
    int num = 10;
    public void method(){
        System.out.println(num);
    }
}
//创建子类继承父类,并定义与父类同名成员变量
public class FieldExtends extends Field {
    int num = 15;
    public void show(){
        System.out.println(num);
    }
    //在子类中定义新的方法,在方法内再定义与父类成员变量同名变量
    public void function(){  
        int num = 5;
        System.out.println(num);
    }
}
//测试类
public class FieldTest {
    public static void main(String[] args) {
        FieldExtends fe = new FieldExtends();
        fe.show();  // 15
        fe.function();   // 5
        fe.method();   //10
    }
}
通过案列可以知道在子类中访问一个变量的顺序
  1. 在子类方法的局部范围找,有就使用
  2. 在子类的成员范围找,有就使用
  3. 在父类成员范围找有就使用
  4. 如果找不到,就报错

super关键字

用法与this关键字类似

  • this 代表对本类对象的引用
  • super 代表父类存储空间的标识(可以理解为对父类对象的引用)

用法

  • 访问成员变量

    this.成员变量
    super.成员变量(访问父类的成员变量,不能访问父类的private变量)
    访问静态成员时,也可以用  父类名.静态成员
    
  • 访问构造方法

    this(...)  super(...)
    
  • 访问成员方法

    this.成员方法()  super.成员方法()
    
案列演示
    //创建父类
    public class FatherSuper {
        int num = 10;
        public FatherSuper(){
            System.out.println("父类构造方法");
        }
        public void show(){
            System.out.println("父类成员方法");
        }
    }
    public class SonSuper extends FatherSuper {
        int num = 15 ;
        public SonSuper(){
            System.out.println("子类构造方法");
        }
        public void method(){
            System.out.println("子类成员方法");
        }
        public void function(){
            int num = 5;                    
            System.out.println(num);        //打印方法内局部变量
            System.out.println(super.num);  //打印父类成员变量
            System.out.println(this.num);   //打印本类成员变量
            super.show();                   //调用父类成员方法
            this.method();                  //调用本类成员方法

        }
    }
    测试类
    public class SuperTest {
        public static void main(String[] args) {
            SonSuper ss = new SonSuper();
            ss.function();
        }
    }

继承中构造方法的关系

  1. 子类中所有的构造方法默认都会访问父类中空参数的构造方法,除非显示使用super/this调用了父类或者本类的其他构造方法。
  2. 在类中对本类或者父类构造方法的调用,只能在构造方法中,不能在实例方法总中调用构造方法(更不能在类方法中调用构造方法)
注意构造方法不能递归调用
    class A {
        public A(int i){
            this(1 ,2);
        }
        public A(int a ,int b){
            this(2);
        }
    }

案例演示

    //创建父类
    public class A {
        public A(){
            System.out.println("父类空参构造");
        }
        public A(String name){
            System.out.println("父类有参构造");
        }
    }
    //创建子类继承父类  
    public class B extends A {
        public B(){           //这一步默认执行了  super();调用了父类空参
            System.out.println("子类空参构造");
        }
        public B(String name){
            super(name);  //调用父类有参构造方法
            System.out.println("子类有参构造");
        }
    }
    测试类
    public class C {
        public static void main(String[] args) {
            B b = new B();
            B b1 = new B("tom");
        } 
    }   
如果父类没有空参构造,子类的构造方法中就必须显式调用父类带参构造super(...);
super(…)或者this(…)必须出现在构造方法第一条语句上否则,就会有父类数据的多次初始化

方法重写

子类中出现和父类中一摸一样的方法声明,称为方法覆盖(Override)或重写(OverWrite)

使用特点

  • 如果方法名不同就调用对应的方法
  • 如果方法名相同最终使用的是自己的(使用子类的引用的时候,在多态情况下,使用父类的引用,则有可能调用的是父类静态方法)

方法重写的应用

  • 当子类需要父类功能时,而功能主体子类有自己的特有的内容时,可以重写父类的方法,这样即沿袭了父类的功能,又定义了特有的内容。
  • 方法重写是多态实现的条件

    //首先创建 一个手机父类
    public class PhoneF {
        public void call(String name){   //定义一个方法
            System.out.println("给"+name+"打电话");
        }
    }
    //在创建一个手机子类继承父类
    public class PhoneS extends PhoneF {
        public void call(String name){   //同名同修饰词   对父类中的方法进行重写
            super.call(name);
            System.out.println("听天气预报");  //在继承父类之后在身再加的打印语句
        }
    }
    //创建测试类
    public class PhoneTest {
        public static void main(String[] args) {
            PhoneS ps = new PhoneS();
            ps.call("Tom");
        }
    }
    

方法重写的注意事项

  • 父类中私有的方法不能被重写,编译报错;
  • 子类重写父类方法时,访问权限不能更低,编译报错;
  • 子类重写父类方法时,返回值类型可以相同,也可以是父类返回值的子类型;
  • 父类的实例方法(非静态方法),子类不能重新定义为静态方法
  • 子类中的重写父类方法,按照重写的原则(访问权限不能变小,返回值类型同类或者为子类,方法名相同,形参列表相同);
  • 子类想重写父类的方法,最好是让方法的签名一模一样;

final关键字

final关键字是最终的意思,可以修饰类,成员变量,成员方法 - 修饰类,类不能被继承(不能放在extends后面) - 修饰变量,变量将变成常量只能被赋值一次,不论是在子类还是本类中,都不能被修改(常量一般是大写字母表示 final int ONE = 1) - 修饰方法,方法不能被重写(子类只用使用权,没有修改权)

final修饰局部变量
  • 在方法内部,该变量不可以被改变
  • 在方法声明上,基本类型是这个参数的值不能被改变,引用类型,是这个参数只想的地址值不能被改变

总结

类中非static的final变量(实例final变量)可以在生命的时候赋值,如果声明的时候没有赋值的话,就必须在以下两个地方赋值,一个是构造代码块一个是构造方法中,如果这两个地方没有赋值,编译报错。

如果是static修饰的final变量的话,则只能在两个地方赋值,声明的时候,或者在静态代码块中。

多态(polymorphism)

某一个事物,在不同时刻表现出来的不同状态

在Java中,对一个事物的引用可以分成两种类型,一种是编译时的类型,一种是运行时的类型。 编译时的类型指的是声明这个变量时指定的类型;运行时类型指的是实际赋给这个变量的对象的类型。

    若Student类继承自Person类,则下面的写法是正确的
    Person p = new Student();
注意

父类引用不能调用子类特有的方法,因为使用了父类的引用,就代表站在父类角度来看待当前子类对象,只能看到从父类继承而来的特性,或者是子类重写的父类的方法。成员变量没有多态性,只能看到父类的成员变量。

多态中的成员访问特点

  • 成员变量 编译看左边,运行看右边;
  • 构造方法 创建子类对象的时候,访问父类的构造方法,对父类的数据进行初始化
  • 成员方法 便宜看左边,运行看右边
  • 静态方法 编译看左边,运行看右边

多态的前提条件

  • 有继承关系
  • 有方法重写
  • 有父类引用指向子类对象

多态中成员访问特点

成员变量和静态方法没有多态性 只有被子类重写的成员方法才有多态性

Day 08Java基础学习笔记