1.继承的引入

在生活中,有形形色色的继承,最常见的继承是一个家族财产的继承关系,儿子可以继承父亲的财产。在Java中也存在继承关系,Java中的继承是子类可以继承父类的成员(成员变量与成员方法)。
Java中继承关系的语法为:
用extends关键字来进行继承,让类与类之间产生继承关系。

下面我们先用生活中一个简单的例子来引入Java中的继承关系:
我们知道动物里面有很多类别,例如猫,狗之类的,都有一些共性的东西。例如吃饭,睡觉等。那么我们就可以将这些共性的东西向上抽取到Animal这个类中,在子类中直接就会继承,可以大量的减少代码量,实现代码的实用性与维护性。

//父类:
public class Anmial {
    //将子类共有的东西向上抽取到父类,以实现代码的复用性与维护性
    public String name;
    public int age;
    public void eat(){
        System.out.println("吃饭");
    }
    public void sleep(){
        System.out.println("睡觉");
    }
}
//子类:
public class Cat extends  Anmial {
    //子类中可以保留自己特有的方法
    public void catchMouse(){
        System.out.println("猫抓老鼠");
    }
}
public class Dog extends Anmial {
    public void lookDoor(){
        System.out.println("小狗看门");
    }
}
//测试类:
public class MyTest {
    public static void main(String[] args) {
        Cat cat = new Cat();
        cat.name="胖橘";
        cat.age=3;
        System.out.println(cat.name);
        System.out.println(cat.age);
        cat.eat();
        cat.sleep();
        cat.catchMouse();
        System.out.println("================");
        Dog dog = new Dog();
        dog.name="小强";
        dog.age=5;
        System.out.println(dog.name);
        System.out.println(dog.age);
        dog.eat();
        dog.sleep();
        dog.lookDoor();
    }
}


运行的结果为:
胖橘
3
吃饭
睡觉
猫抓老鼠
================
小强
5
吃饭
睡觉
小狗看门

  • 继承的利弊分析

继承的好处

继承的弊端

提高了代码的复用性与维护性

让类与类之间产生了依赖的关系,增加了耦合性

继承是多态的前提,有了继承才能谈及多态

当依赖关系太过于紧密时,会违背软件的设计原则:高内聚,低耦合

让类与类之间有了继承关系

耦合性过高时,不利于软件后期的维护和扩展


2.继承的特点

  • Java中继承语法:Java中只支持单继承,不支持多继承。可以支持多层继承
  • Java中子类只能继承父类的非私有成员。私有成员,子类不能继承。
  • 构造方法不参与继承
  • 静态方法可以继承
//Object 是我们Java继承体系当中的顶层父类
//所有的类都是直接或间接继承自他。
public class MyTest2 extends Object{
    public static void main(String[] args) {
        /*Java中继承的语法特点
        1.Java中只支持单继承
        一个子类只能有一个父类,
        一个父类是可以有多个子类。
        不支持多继承。可以支持多层继承。
        2.子类只能继承父类非私有的成员。私有的成员,子类不能继承。
        3.构造方法,不参与继承。构造方法,在你初始化子类对象时,
            会调用父类构造来完成父类数据的初始化。
        4.静态方法也是可以继承的。*/

        D d = new D();
        d.hehe();

        D.hehe();


    }
}

class A extends Object{
//子类不能继承父类中的私有变量
   private int a=10;
    public void a(){

    }
}

class B extends A{
    int b=100;
    //子类不能继承父类的私有方法
    private void b(){}
}

class C extends B{
    int c=70;
    public void c(){}
    public static void hehe(){

    }
}
class D extends C{

}
}

  • 当我们创建子类对象时,会先调用父类的构造方法的原因:当我们去创建子类对象时,子类会继承父类的一些数据,会先完成父类数据的初始化,然后初始化自己的数据
public class MyTest {
    public static void main(String[] args) {
        /*当我们创建子类对象时,会先调用父类的构造方法,为什么?
        当我们去创建子类对象时,子类要继承父类的一些数据甚至还要去使用,
        那如果子类的数据没有完成初始化,子类能继承吗?肯定不行。
        所以当我们创建子类对象时,应该先完成父类数据的初始化,
        然后在初始自己的数据。
         */
        /*创建子类成员变量时,会先调用父类的构造方法,
        然后在调用子类构造方法
         */
        Zi zi = new Zi();

    }
}

class Fu extends Object{
    int a=20;
   public Fu(){
       super();
       System.out.println("父类的构造方法执行了");
   }
}

class Zi extends Fu{
    public Zi(){
        super(); //调用父类的空参构造
        System.out.println("子类的构造方法执行了");
    }
}

3.this 、super关键字

super关键字

super:表示父类的空间标识,可以简单理解为父类的一个引用,使用super可以调用父类的数据

super.变量名:调用父类的变量

super.方法名():调用父类的方法

super():调用父类的空参构造

super(有参):调用父类的有参构造

this关键字

this:表示本类的一个引用,谁调用这个方法,方法中的this就代表谁

this.变量名:调用本类的变量

this.方法名():调用本类的方法

this():调用本类的空参构造

this(有参):调用本类的有参构造

public class MyTest {
    public static void main(String[] args) {
        ZZ zz = new ZZ();
        zz.ziHeHe(140);
    }
}
class FF{
    int num=20;
    private int bb=250;
    public int cc=1405;

    public void fuShow(){
        System.out.println(num+"父类的方法");
    }
    public int  getBB(){
        return this.bb;
    }
}
class ZZ extends FF{
    int z=100;
    int num=800;

    public void ziHeHe(int num) {
        /*
          变量访问的就近原则。
        * 当我们在方法中访问一个变量的时候,先从局部范围找,
          找到就使用,如果没找到,去本类的成员范围找,找到就使用,
        * 如果没找到,会去父类的成员范围找,找到就使用
        * 局部变量和成员变量以及父类的成员变量重名问题,
          可以使用  this super 来区分
        * */

        System.out.println(num);//140
        System.out.println(this.num);//800
        System.out.println(super.num);//20
        System.out.println(cc);
    }
    public void ziShow(){
        System.out.println(this.z);
        System.out.println(this.num);
        System.out.println(super.num);
        this.fuShow();
        super.fuShow();
       // System.out.println(super.bb); 报错
        int bb = this.getBB();
        int bb1 = super.getBB();
        System.out.println(bb);
        System.out.println(bb1);

    }


}

4.方法重写

在继承关系中,当子类与父类存在一模一样的方法,可以进行方法重写。

  • 方法重写:子类中出现与父类一模一样的方法(方法名,参数列表以及返回值的类型),也称为方法重写或者方法覆盖。
  • 子类会覆盖父类的方法,实际运行时,运行的是子类重写过的方法
  • 方法重写的机制:一个父类会有多个子类,父类里面抽取的是所有子类共性的功能,但每个子类对于共性功能的具体实现,存在差异。
  • 子类方法重写的目的:就是对父类功能实现不满意,想要根据自己的需求,进行重写与扩展

4.1 方法重写(功能重写)

//父类
public class Animal {
	//对于子类中共性功能进行提取
    public void eat() {
        System.out.println("吃饭");
    }

    public void sleep() {
        System.out.println("睡觉");
    }
}
//子类
public class Cat extends Animal{
    public void catchMouse(){
        System.out.println("抓老鼠");
    }

    public void eat(){
        System.out.println("猫爱吃小鱼干");
    }

    public void sleep(){
        System.out.println("猫喜欢白天睡觉");
    }

}
public class Dog extends Animal{
    public void lookDoor(){
        System.out.println("狗看门");
    }

    public void eat(){
        System.out.println("狗吃骨头");
    }
}
//测试类
public class MyTest {
    public static void main(String[] args) {
        Cat cat = new Cat();
        cat.eat();
        cat.sleep();
        System.out.println("=====================");
        Dog dog = new Dog();
        dog.sleep();
        dog.eat();
    }
}


输出的结果为:

猫爱吃小鱼干
猫喜欢白天睡觉
=====================
睡觉
狗吃骨头

4.2 方法重写(功能扩展)

  • 对于某部分代码满意,还想扩展,也可以通过方法重写来实现
//父类:手机类
public class Phone {
    public Phone() {
        System.out.println("构造方法执行了");
    }

    public void call() {
        System.out.println("打电话 ");
    }

    public void send() {
        System.out.println("发短信");
    }
}
//子类:Iphone手机
public class Iphone extends Phone{
    public void palyGame(){
        System.out.println("玩游戏");
    }

    @Override //注解   
    // @Override 这个注解的作用,是用来检测此方法是不是重写父类的方法
    public void call() {
        super.call(); //沿袭父类的功能
        System.out.println("高清视频通话");
    }
}
//测试类:
public class MyTest {
    public static void main(String[] args) {
        Iphone iphone = new Iphone();
        iphone.call();
        iphone.send();
        iphone.palyGame();
    }
}

输出的结果为:
打电话
高清视频电话
发短信

4.2 方法重写的注意事项

方法重写的注意事项

父类的私有方法,子类不能进行重写(因为父类的私有方法,子类都不能继承,就更不能进行重写

父类的构造方法不能重写(构造方法不参与继承)

静态方法不参与重写

子类在重写父类方法时,方法的权限修饰符,不能比父类的低,可以与父类的权限修饰符保持一致(权限:public>protected>缺省>私有)

public class MyTest {
    public static void main(String[] args) {
        //方法重写时,需要注意的事项。
        //1.父类私有的方法,子类不能重写,私有方法,子类不能继承
        //2.父类构造方法,不能重写,构造方法没有继承一说。
        //3.子类在重写父类方法时,方法的权限修饰符,
        // 不能比父类的低可以跟父类的一样,或者比父类的高
        //public>protected>缺省的>private
        //4.静态方法,不参与重写

        Father.haha();

        Son son = new Son();
        son.haha();
        Son.haha();

    }

}

class Father{
    private void show(){

    }

    protected void hehe(){

    }
    public static void haha(){
        System.out.println("父类的静态方法");
    }

}
class Son extends Father{
    @Override
    protected void hehe() {

    }

    //@Override  静态方法不参与重写
    public static void haha() {
        System.out.println("子类的静态方法");
    }

}

5.final关键字

  • final关键字是最终的意思,可以修饰类,变量,成员方法。

final修饰特点

修饰类: 被修饰类不能被继承

修饰方法: 被修饰的方法不能被重写

修饰变量: 被修饰的变量不能被重新赋值,因为这个量其实是一个常量

public class MyTest {
    //公共的静态常量
 public static final double AA=3.14; //自定义常量
    public static void main(String[] args) {
        //final 最终的,可以修饰成员变量,局部变量,形参,也可以修饰方法,也可以修类
        //修饰变量,此变量为一个常量,既然为常量,常量的值就不能再次被改变。
        //自定义常量,常量名一般建议大写
        final int NUM=100;
              //NUM=200; //重新赋值,报错,常量不能再次赋值
        System.out.println(NUM);
        double aa = MyTest.AA;
        System.out.println(aa);

        BB bb = new BB();
        System.out.println(bb);
           bb=new BB();
        System.out.println(bb);

        System.out.println("============================");
       final BB bb1 = new BB();
        //final修饰的是基本类型,指的是值不能再次被改变。
        //final修饰的是引用数据类型,指的是这个地址值,不能再次被改变
        //final修饰方法,此方法,不能被子类重写,但是可以让子类继承去用
        CC cc = new CC();
        cc.show();
        //final 修饰类,此类不能被继承
      
    }
}

class BB{
   public final void show(){
       System.out.println("父类的final修饰的方法,子类不能重写,子类可以继承");
   }
}

class CC extends BB{
 /*   @Override
    public void show() {

    }*/
}
//final修饰类,此类不能被继承
 final class EE{

}
/*
class FF extends EE{

}
*/