Bootstrap

Java OOP进阶:继承,抽象类、接口、多态

先分享一张弔图

继承

继承是什么?

继承就是子类继承父类的-8特征和行为,使得子类对象(实例)具有父类的实例域和方法,或子类从父类继承方法,使得子类具有父类相同的行为。

继承是对现实生活中的“分类”概念的一种模拟。 

比如说写了一个叫Animal的类,里面的成员是动物的共性。再写一个叫Dog的类来继承Animal类。狗拥有动物的一切基本特性,但同时又拥有自己的独特的特性,这就是“继承”关系的重要特性:通常简称为“IS_A”关系。

Java的继承语法:

1. public:外界可自由访问
2. private:外界不可访问
3. protected:同一包中的子类都可以访问,另一包中的子类(派生于同一个父类)也可以访问
4. default:如果不指明任何权限,则默认同一包中的类可以访问

class 父类 {

        ……
}
 
class 子类 extends 父类 {

        ……
}

子类自动拥有父类声明为public和protected的成员。

 继承类型

  • Java 的继承是单继承,但是可以多重继承,单继承就是一个子类只能继承一个父类,多重继承就是,例如 B 类继承 A 类,C 类继承 B 类,所以按照关系就是 B 类是 C 类的父类,A 类是 B 类的父类,这是 Java 继承区别于 C++ 继承的一个特性。


class Grandparent {

    public Grandparent() {
        System.out.println("GrandParent Created.");
    }

    public Grandparent(String string) {
        System.out.println("GrandParent Created.String:" + string);
    }
}

class Parent extends Grandparent {

    public Parent() {
		//请试着先后取消这两句的注释,看看会有什么事情发生
        super("Hello.Grandparent.");
        System.out.println("Parent Created");
        //super("Hello.Grandparent.");
    }
}

class Child extends Parent {

    public Child() {
        System.out.println("Child Created");
    }
}

public class TestInherits {

    public static void main(String args[]) {
        Child c = new Child();
    }
}

创建子类对象时,基类的构造方法会被自动地调用。而且是从顶向下调用。

如果要使用super关键字调用基类的特定构造方法,则必须是子类构造方法中的第一条语句。

因为,依据继承的特性,子类代码是可以访问父类成员的,在某些情况下,子类字段的初始化可能需要访问基类的字段,因此,为保证子类字段在任何情况下都能顺利地被初始化,先初始化基类的字段,让其有确定的值,是合理的。

关键字

super 关键字:

我们可以通过 super 关键字来实现对父类成员的访问,用来引用当前对象的父类。

java文件被编译成class文件时,在子类的所有构造函数中的第一行(第一个语句)会默认自动添加 super() 语句,在执行子类的构造函数前,总是会先执行父类中的构造函数。

在编写代码要注意:

  •  1.如果父类中不含 默认构造函数(就是 类名() ),那么子类中的super()语句就会执行失败,系统就会报错。一般 默认构造函数 编译时会自动添加,但如果类中已经有一个构造函数时,就不会添加。
  •  2.执行父类构造函数的语句只能放在函数内语句的首句,不然会报错。

this 关键字:

指向自己的引用,引用当前对象,即它所在的方法或构造函数所属的对象实例。

class Animal {
    void eat() {
        System.out.println("animal : eat");
    }
}
 
class Dog extends Animal {
    void eat() {
        System.out.println("dog : eat");
    }
    void eatTest() {
        this.eat();   // this 调用自己的方法
        super.eat();  // super 调用父类方法
    }
}
 
public class Test {
    public static void main(String[] args) {
        Animal a = new Animal();
        a.eat();
        Dog d = new Dog();
        d.eatTest();
    }
}

final 关键字

前面的文章中提到过,final 可以用来修饰变量(包括类属性、对象属性、局部变量和形参)、方法(包括类方法和对象方法)和类。

使用 final 关键字声明类,就是把类定义定义为最终类,不能被继承,或者用于修饰方法,该方法不能被子类重写:

利用final,我们可以设计出一种特殊的“只读” 的“不可变类”

“不可变类”可以方便和安全地用于多线程环境中,访问它们可以不用加锁,因而能提供较高的性能。

final 的作用随着所修饰的类型而不同
1、final 修饰类中的属性或者变量

无论属性是基本类型还是引用类型,final 所起的作用都是变量里面存放的"值"不能变。

这个值,对于基本类型来说,变量里面放的就是实实在在的值,如 1,"abc" 等。

而引用类型变量里面放的是个地址,所以用 final 修饰引用类型变量指的是它里面的地址不能变,并不是说这个地址所指向的对象或数组的内容不可以变,这个一定要注意。

例如:类中有一个属性是 final Person p=new Person("name"); 那么你不能对 p 进行重新赋值,但是可以改变 p 里面属性的值 p.setName('newName');

final 修饰属性,声明变量时可以不赋值,而且一旦赋值就不能被修改了。对 final 属性可以在三个地方赋值:声明时、初始化块中、构造方法中,总之一定要赋值。

2、final修饰类中的方法

作用:可以被继承,但继承后不能被重写。

3、final修饰类

作用:类不可以被继承

构造器

子类是不继承父类的构造器(构造方法或者构造函数)的,它只是调用(隐式或显式)。如果父类的构造器带有参数,则必须在子类的构造器中显式地通过 super 关键字调用父类的构造器并配以适当的参数列表。

如果父类构造器没有参数,则在子类的构造器中不需要使用 super 关键字调用父类构造器,系统会自动调用父类的无参构造器。

class SuperClass {
    private int n;
 
    // 无参数构造器
    public SuperClass() {
        System.out.println("SuperClass()");
    }
 
    // 带参数构造器
    public SuperClass(int n) {
        System.out.println("SuperClass(int n)");
        this.n = n;
    }
}
 
// SubClass 类继承
class SubClass extends SuperClass {
    private int n;
 
    // 无参数构造器,自动调用父类的无参数构造器
    public SubClass() {
        System.out.println("SubClass()");
    }
 
    // 带参数构造器,调用父类中带有参数的构造器
    public SubClass(int n) {
        super(300);
        System.out.println("SubClass(int n): " + n);
        this.n = n;
    }
}
 
// SubClass2 类继承
class SubClass2 extends SuperClass {
    private int n;
 
    // 无参数构造器,调用父类中带有参数的构造器
    public SubClass2() {
        super(300);
        System.out.println("SubClass2()");
    }
 
    // 带参数构造器,自动调用父类的无参数构造器
    public SubClass2(int n) {
        System.out.println("SubClass2(int n): " + n);
        this.n = n;
    }
}
 
public class TestSuperSub {
    public static void main(String[] args) {
        System.out.println("------SubClass 类继承------");
        SubClass sc1 = new SubClass();
        SubClass sc2 = new SubClass(100);
 
        System.out.println("------SubClass2 类继承------");
        SubClass2 sc3 = new SubClass2();
        SubClass2 sc4 = new SubClass2(200);
    }
}

子类与父类方法

在继承关系中,在调用函数(方法)或者类中的成员变量时,JVM(JAVA虚拟机)会先检测当前的类(也就是子类)是否含有该函数或者成员变量,如果有,就执行子类中的,如果没有才会执行父类中的。在子类中,若要调用父类中被重写(或覆盖)的方法,可以使用super关键字。

由于Java并未对子类方法的命名做过多的限制,因此,子类与父类各自定义的方法之间,可以出现以下三种情况:

1. 扩充(Extend):子类定义的方法,父类没有同名方法
2. 覆盖/重写(Override):子类父类定义了完全一样的方法
3. 重载(Overload):子类有父类的同名方法,但两者的参数类型或参数数目不一样

Java“方法覆盖/重写”的语法规则

1. 覆盖/重写方法的允许访问范围不能小于原方法。
2. 覆盖/重写方法所抛出的异常不能比原方法更多。
3. 声明为final方法不允许覆盖/重写。例如,Object的getClass()方法不能覆盖/重写。
4. 不能覆盖/重写静态方法。

public class ShapeAndRectangle {
}

class Shape{
    private final int top;
    private final int left;
    //基类的构造方法
    public Shape(int top, int left) {
        this.top = top;
        this.left = left;
    }
}

class Rectangle extends Shape{
    private final int width;
    private final int height;

    public Rectangle(int width,int height){
        //调用本类的构造方法
        this(0,0,width,height);
    }

    public Rectangle(int top, int left,
                     int width, int height) {
        super(top, left); //调用父类的构造方法
        this.width = width;
        this.height = height;
    }
}

顶层基类Object

在Java中,所有的类都直接或间接地派生自最顶层的基类Object,此类定义了一些方法,如图所示。

所有的类都是继承于 java.lang.Object,当一个类没有继承的关键字,则默认继承 Object(这个类在 java.lang 包中,所以不需要 import)祖先类。也就是说是所有类的根父类,这个可以运用于参数的传递。

public class Start
{
    public static void main(String[] args)
    {
        A a=new A();
        B b=new B();
        C c=new C();
        D d=new D();
        speak(a);
        speak(b);
        speak(c);
        speak(d);
    }
// instanceof 关键字是用于比较类与类是否相同,相同返回true,不同返回false
//当你不清楚你需要的参数是什么类型的,可以用Object来代替,Object可以代替任何类
    static void speak(Object obj)
    {
        if(obj instanceof A)//意思是:如果参数是 A 类,那么就执行一下语句
        {
            A aobj=(A)obj;//这里是向下转换,需要强制转换
            aobj.axx();
        }
        else if(obj instanceof B)
        {
            B bobj=(B)obj;
            bobj.bxx();
        }
        else if(obj instanceof C)
        {
            C cobj=(C)obj;
            cobj.cxx();
        }
    }
}
//这里举了四个类,他们的函数都不同,但都是 Object 类的子类
class A
{
    void axx()
    {
        System.out.println("Good morning!");
        System.out.println("This is A");
    }
}

class B
{
    void bxx()
    {
        System.out.println("Holle!");
        System.out.println("This is B");        
    }
}

class C
{
    void cxx()
    {
        System.out.println("Look!");
        System.out.println("This is C");        
    }
}

class D
{
    void dxx()
    {
        System.out.println("Oh!Bad!");
        System.out.println("This is D");        
    }
}


关于“继承”需要注意的点:

1.当子类重写基类的方法时,注意不要彻底改变方法的职责。子类应该是对基类同名方法行为特性的“调整”或“增强”。

2.尽量少写依据变量类型(是父类还是子类)进行判断的条件语句,而应该采用“多态”来简化代码。

3.避免继承的层次的过深,也就是说,从子类到最顶层的基类的“中间”类的个数不要太多,自定义类的继承层次最好限制在三层以下。

4.当类所对应的客观事物之间存在着“是一种(IS_A)”关系时,为这些类之间建立继承关系才是合理的。

5.基类中可以保存一些所有子类都要用的公用代码,但不要仅为“代码重用”而在类型之间建立继承关系,考虑“组合”取代“继承”。

抽象类

在面向对象的概念中,所有的对象都是通过类来描绘的,但是反过来,并不是所有的类都是用来描绘对象的,如果一个类中没有包含足够的信息来描绘一个具体的对象,这样的类就是抽象类。

抽象类除了不能实例化对象之外,类的其它功能依然存在,成员变量、成员方法和构造方法的访问方式和普通类一样。

由于抽象类不能实例化对象,所以抽象类必须被继承,才能被使用。也是因为这个原因,通常在设计阶段决定要不要设计抽象类。

父类包含了子类集合的常见的方法,但是由于父类本身是抽象的,所以不能使用这些方法。

在 Java 中抽象类表示的是一种继承关系,一个类只能继承一个抽象类,而一个类却可以实现多个接口。

抽象类的特点:

▪ 有abstract修饰的类称为“抽象类”,它只定义了什么方法应该存在,不能直接创建对象。你必须从它派生出一个子类,并在子类中实现其未实现的方法之后,才能使用new关键字创建它的实例。
▪ 在方法前加上abstract就形成抽象方法,这种类型的方法,只有方法声明,没有实现代码。

当然,抽象类中可以包含非抽象方法,也可以象普通类一样,定义字段。

包含抽象方法的类一定是抽象类,但抽象类中的方法不一定是抽象方法。

抽象方法

如果你想设计这样一个类,该类包含一个特别的成员方法,该方法的具体实现由它的子类确定,那么你可以在父类中声明该方法为抽象方法。

Abstract 关键字同样可以用来声明抽象方法,抽象方法只包含一个方法名,而没有方法体。

抽象方法没有定义,方法名后面直接跟一个分号,而不是花括号。

声明抽象方法会造成以下两个结果:

  • 如果一个类包含抽象方法,那么该类必须是抽象类。
  • 任何子类必须重写父类的抽象方法,或者声明自身为抽象类。

继承抽象方法的子类必须重写该方法。否则,该子类也必须声明为抽象类。最终,必须有子类实现该抽象方法,否则,从最初的父类到最终的子类都不能用来实例化对象。

使用抽象类

▪ 抽象类不能创建对象,一般用它来引用子类对象。

就像下面例子中的:Pet d = new Dog();

抽象类 抽象类变量 = new 派生自抽象类的具体子类();

在现实生活中,属于同一种类的事物,在特定的方面,表现出不同的特点,比如狗和猫都会叫,但两者的叫声是不同的。

• 由于狗和猫都是一种(IS_A)宠物,所以设计一个Pet类,让Dog和Cat成为它的子类。

• 由于狗和猫都会叫,所以给Pet类添加一个speak()方法,又由于并不存在一种“统一”的宠物叫声,而狗和猫的叫声又不一样,所以,应该让Pet.speak()方法成为抽象的。Pet作为抽象类,而Cat和Dog重写基类的speak()方法。

public class UseAbstractClass {

    public static void main(String[] args) {
       Pet d = new Dog();
       //汪汪汪~~~
       d.speak();
       //喵喵喵~~~
       Pet c = new Cat();
       c.speak();
    }
}

abstract class Pet {
    abstract void speak();
}

class Dog extends Pet {

    @Override
    void speak() {
        System.out.println("汪汪汪~~~");
    }
}

class Cat extends Pet {

    @Override
    void speak() {
        System.out.println("喵喵喵~~~");
    }
}

tips:编译器使用技巧

当IntelliJ检测到你正在编写的类派生自一个抽象类时,会打上红色波浪线,提醒你需要实现基类所定义的抽象方法,并且提供了自动生成方法框架代码的快捷功能。

接口

接口(英文:Interface),在JAVA编程语言中是一个抽象类型,是抽象方法的集合,接口通常以interface来声明。一个类通过继承接口的方式,从而来继承接口的抽象方法。

接口并不是类,编写接口的方式和类很相似,但是它们属于不同的概念。类描述对象的属性和方法。接口则包含类要实现的方法。

除非实现接口的类是抽象类,否则该类要定义接口中的所有方法。

接口无法被实例化,但是可以被实现。一个实现接口的类,必须实现接口内所描述的所有方法,否则就必须声明为抽象类。另外,在 Java 中,接口类型可用来声明一个变量,他们可以成为一个空指针,或是被绑定在一个以此接口实现的对象。

接口与类的关系

接口与类相似点:

  • 一个接口可以有多个方法。
  • 接口文件保存在 .java 结尾的文件中,文件名使用接口名。
  • 接口的字节码文件保存在 .class 结尾的文件中。
  • 接口相应的字节码文件必须在与包名称相匹配的目录结构中。

接口与类的区别:

  • 接口不能用于实例化对象。
  • 接口没有构造方法。
  • 接口中所有的方法必须是抽象方法,Java 8 之后 接口中可以使用 default 关键字修饰的非抽象方法。
  • 接口不能包含成员变量,除了 static 和 final 变量。
  • 接口不是被类继承了,而是要被类实现。
  • 接口支持多继承。

接口的语法特性

定义一个接口,采用关键字interface;某个类实现某个接口,采用关键字implements。

接口的成员函数自动成为public的,数据成员自动成为static和final的。

如果接口不声明为public的,则自动变为package的。

一个类可以同时实现多个接口,但只能有一个基类。基类必须写在第一位,后面跟着的是它所实现的多个接口。

▪ 可以通过继承接口来扩充已有接口,并形成一个新的接口。下面的代码块中,[ ]内的可以不加

[可见度] interface 接口名称 [extends 其他的接口名] {
        // 声明变量
        // 抽象方法
}

Interface关键字用来声明一个接口。下面是接口声明的一个简单例子。

/* 文件名 : NameOfInterface.java */
import java.lang.*;
//引入包
 
public interface NameOfInterface
{
   //任何类型 final, static 字段
   //抽象方法
}

接口有以下特性:

  • 接口是隐式抽象的,当声明一个接口的时候,不必使用abstract关键字。
  • 接口中每一个方法也是隐式抽象的,声明时同样不需要abstract关键字。
  • 接口中的方法都是公有的。

类调用接口

class 调用接口的类名 implements 接口名{

        //对接口内的抽象方法具体实现

}

接口的使用

接口类型 接口类型的变量 = new 实现了接口的具体类型();

例子

因为“鸭子是一种(IS_A)鸟,又是一种(IS_A)食物”,所以我们可以创建三个类:Bird代表鸟,Food代表食物,Duck代表鸭子,让Duck派生自Bird和Food。

但是由于Java不支持多继承(但是c++支持)︿( ̄︶ ̄)︿,所以使用“接口(interface)”被用来抽象对象的行为特性或外部特征。

Duck类派生自Bird类,实现IFood和ISwim接口,鸭子是一种鸟(用“继承”特性表达),会游泳,同时又是一种食物(用它所实现的“接口”表达)

代码:注意注释

public class DuckAndBird {
    public static void main(String[] args) {
        Duck duck = new Duck();
        duck.fly();
        //Duck对象实现了IFood和ISwim接口,
        //所以可以赋值给相应的变量
        IFood food = duck;
        //分别用下面三行替换上面这行,第一行可以正常运行,第二,三行则不行。
        //这是因为Duck类中包括了IFood接口,所以可以调用接口内的方法。但是如果换成Bird和ISwim则不行,
        // 因为这两个类(接口)中没有包含 .cook() 方法
        //Duck food = duck;
        //Bird food = duck;
        //ISwim food = duck;
        food.cook();
        ISwim swimAnimal = duck;
        swimAnimal.swim();
    }
}

//所有食物应该实现这个接口
interface IFood {
    void cook();
}
//所有的水鸟,应该实现此接口
interface ISwim {
    void swim();
}

abstract class Bird {
    //鸟有翅膀,会飞!
    abstract void fly();
}
//鸭子是一种水鸟,也是一种食物
class Duck extends Bird
        implements ISwim, IFood {

    @Override
    public void cook() {
        System.out.println("我能做成烤鸭!");
    }

    @Override
    public void swim() {
        System.out.println("我能游泳!");
    }

    @Override
    void fly() {
        System.out.println("我会飞!");
    }
}

利用接口定义常量

public interface ArrayBound {
         public static final int LOWBOUND = 1;
         public static final int UPBOUND = 100;
}

以这种方式定义的常量,说明这些常量需要被实现这个接口的类所使用。请注意要为它们设计出有意义的名字,以便保证代码的可读性。

只要一个类声明实现了这个接口,就可以直接使用这些常量。

定义在接口中的常量必须被初始化

例子:就这样使用

public interface ArrayBound {  
    public static final int LOWBOUND = 1;    // 数组的下界  
    public static final int UPBOUND = 100;   // 数组的上界  
}

public class ConstantPrinter implements ArrayBound {  

    public void printBounds() {  
        System.out.println("下界: " + LOWBOUND);  
        System.out.println("上界: " + UPBOUND);  
    }  

    public static void main(String[] args) {  
        ConstantPrinter printer = new ConstantPrinter();  
        printer.printBounds();  // 打印界限值  
    }  
}  

函数式接口

JDK 8中引入了一种“函数式”接口,这种接口只定义有一个公有方法,使用@FunctionalInterface注解加以标注:

@FunctionalInterface
interface MyFunctionInterface {
    void process();
}


这种类型的接口,主要用于“函数式编程”场景,使用它定义的变量,可以接收一个Lambda表达式。

import java.time.LocalDate;

public class UseFunctionalInterface {
    public static void main(String[] args) {
        MyFunctionInterface obj = () -> { //函数式接口变量,可以引用一个Lambda表达式。
            var info = """
                    工作完成于:%s
                    """.formatted(LocalDate.now());
            System.out.println(info);
        };
        obj.process();
    }
}

@FunctionalInterface
interface MyFunctionInterface {
    void process();
}

有关Lambda表达式的内容,之后再介绍。(疯狂挖坑)

JDK9引入的新特性

  • 接口中的私有方法,主要用于封装一些接口的默认方法(可以有多个)需要调用的公用代码。
public interface MyInterface {
    //JDK 9: 接口中可以定义私有成员,它可以被接口的默认方法调用
    private void secret() {
        System.out.println("Secret Method defined in interface");
    }
    //接口的默认方法
    default void defaultMethod(){
        secret();
        System.out.println("defaultMethod()");
    }
    void func();
}
  • 实现了接口的具体类型,如果没有重写,就直接“继承”接口所定义的默认方法。
public class MyClass implements MyInterface {
    @Override
    public void func() {
        System.out.println("MyClass.func()");
    }

    public static void main(String[] args) {
        MyInterface obj = new MyClass();
        obj.defaultMethod();   //直接使用接口所定义的默认方法
        obj.func();
    }
}

interface MyInterface {
    //JDK 9: 接口中可以定义私有成员,它可以被接口的默认方法调用
    private void secret() {
        System.out.println("Secret Method defined in interface");
    }
    //接口的默认方法
    default void defaultMethod(){
        secret();
        System.out.println("defaultMethod()");
    }
    void func();
}

MyClass 实现了 MyInterface 接口,并提供了 func() 方法的具体实现。因为 MyClass 没有重写 defaultMethod(),所以它自动“继承”了接口中定义的默认实现。当调用obj.defaultMethod()时,该方法中的逻辑将被执行。

多态

在面向对象理论中,多态性的定义是:同一操作,作用于不同的类的实例,不同的类将进行不同的解释,最后产生不同的结果。

利用多态特性,看上去完全一样的一条语句,在不同的运行环境中,却可以产生不同的运行结果。

多态代码的最本质特征就是——父类(或接口)变量可以引用子类(或实现了接口的类)对象。换句话说:子类对象可以被当成基类对象使用。其典型代码如下所示:

//父类变量引用子类对象 
Parent p = new Child();


//MyClass类实现了IMyInterface接口
IMyInterface obj = new MyClass();

这个特性其实在前面的例子代码中也写了,只是没有展开来细说。

作用:

  • 总是可以让更一般的对象引用更具体化的对象。
  • Java类库的最顶层类是Object。因此每个对象都可以赋值给Object变量。但是这样只能调用Object类中的方法,比如equal , getClass等

子类与基类变量间的赋值

向上转型

子类对象赋值给基类引用是一种常见的多态性特征。通过这种方式,可以使用基类类型的引用来操作子类对象。具体来说,当你将子类对象赋值给基类引用时,Java会自动进行类型转换(向上转型)。这被称为向上转型(Upcasting)

向上转型是安全的,不会导致运行时错误,但在使用基类引用时只能访问基类的方法和属性。

向下转型

  • 基类对象要赋给子类对象变量,必须执行类型强制转换,其语法是:

子类对象变量=(子类名)基类对象变量;

  • 如果需要访问子类特有的方法,需要进行向下转型(Downcasting)。向下转型时需要小心操作,因为只有在基类引用实际上指向的对象是所转换的子类的实例时,向下转型才是安全的。换句话说,如果你尝试将一个基类引用转换为不兼容的子类类型,就会抛出异常。

父类引用指向不同子类的实例: 如果父类引用实际指向一个不同子类的实例,而你尝试将其转换为不相干的子类。

class Animal {}  
class Dog extends Animal {}  
class Cat extends Animal {}  

public class Test {  
    public static void main(String[] args) {  
        Animal myAnimal = new Cat(); // myAnimal指向Cat实例  
        Dog myDog = (Dog) myAnimal; // 这个向下转型会抛出ClassCastException  
    }  
}

null 值: 尝试将 null 值进行向下转型不会抛出 ClassCastException,但如果你在访问该引用的成员时,就会遇到 NullPointerException

使用 instanceof 检查: 在进行向下转型之前,最好使用 instanceof 关键字检查对象的实际类型。这可以确保你不会尝试将一个不兼容的类进行转换。

为了避免 ClassCastException,可以使用 instanceof 来检查类型:

class Animal {}  
class Dog extends Animal {  
    void bark() {  
        System.out.println("Bark");  
    }  
}  

public class Test {  
    public static void main(String[] args) {  
        Animal myAnimal = new Dog(); // 向上转型  

        // 使用 instanceof 进行检查  
        if (myAnimal instanceof Dog) {  
            Dog myDog = (Dog) myAnimal; // 安全的向下转型  
            myDog.bark(); // 调用 Dog 类的方法  
        } else {  
            System.out.println("myAnimal is not an instance of Dog");  
        }  
    }  
}

子类和父类出现同名成员

同名方法

先给一个例子来测试:


public class ParentChildTest {
    public static void main(String[] args) {
        var parent = new ParentClass();
        parent.printValue();
        var child = new ChildClass();
        child.printValue();

        parent = child;
        parent.printValue();

        parent.myValue++;
        parent.printValue();

        ((ChildClass) parent).myValue++;
        parent.printValue();
    }
}

class ParentClass {
    public int myValue = 100;

    public void printValue() {
        System.out.println("Parent.printValue(),myValue=" + myValue);
    }
}

class ChildClass extends ParentClass {
    public int myValue = 200;

    public void printValue() {
        System.out.println("Child.printValue(),myValue=" + myValue);
    }
}

这个代码的运行结果是

当子类与父类拥有一样的方法,并且让一个父类变量引用一个子类对象时,到底调用哪个方法,由引用对象的“真实”类型所决定,这就是说:引用的对象是子类型的,它就调用子类型的方法,是父类型的,它就调用父类型的方法

同名同类型的字段

如果子类与父类有相同的字段,则子类中的字段会代替或隐藏父类的字段,子类方法中访问的是子类中的字段(而不是父类中的字段)。如果子类方法确实想访问父类中被隐藏的同名字段,可以用之前提到过的super关键字来访问它。

然而,如果子类被当作父类使用,则通过子类访问的字段是父类的。

class Parent
        {
	public int value=100;
    public void Introduce(){
        System.out.println("I'm father");
    }

}
class Son extends Parent{
	public int value=101;
     public void Introduce(){
        System.out.println("I'm son");
    }
}

class Daughter extends Parent{
	public int value=102;
 public void Introduce(){
        System.out.println("I'm daughter");
    }
}
public class TestPolymorphism {

    public static void main(String args[]){
        Parent p=new Parent();
        p.Introduce();
        System.out.println(p.value);
        p=new Son();
        p.Introduce();
        System.out.println(p.value);
        p=new Daughter();
        p.Introduce();
        System.out.println(p.value);

    }
}

c++与Java的面向对象语法比较

由于是先学的c++的面向对象,所以在此比较一下两种语言的区别

1. Java 虚函数

虚函数的存在是为了多态。

C++ 中普通成员函数加上 virtual 关键字就成为虚函数。

Java 中其实没有虚函数的概念,它的普通函数就相当于 C++ 的虚函数,动态绑定是 Java 的默认行为。如果 Java 中不希望某个函数具有虚函数特性,可以加上 final 关键字变成非虚函数。

PS: 其实 C++ 和 Java 在虚函数的观点大同小异,异曲同工罢了。

2. Java抽象函数(纯虚函数)

抽象函数或者说是纯虚函数的存在是为了定义接口。

C++ 中纯虚函数形式为:

virtual void print() = 0;

Java 中纯虚函数形式为:

abstract void print();

PS: 在抽象函数方面 C++ 和 Java 还是换汤不换药。

3. Java 抽象类

抽象类的存在是因为父类中既包括子类共性函数的具体定义,也包括需要子类各自实现的函数接口。抽象类中可以有数据成员和非抽象方法。

C++ 中抽象类只需要包括纯虚函数,既是一个抽象类。如果仅仅包括虚函数,不能定义为抽象类,因为类中其实没有抽象的概念。

Java 抽象类是用 abstract 修饰声明的类。

PS: 抽象类其实是一个半虚半实的东西,可以全部为虚,这时候变成接口。

4. Java 接口

接口的存在是为了形成一种规约。接口中不能有普通成员变量,也不能具有非纯虚函数。

C++ 中接口其实就是全虚基类。

Java 中接口是用 interface 修饰的类。

PS: 接口就是虚到极点的抽象类。

总结来说就是这样:

C++ 虚函数    ==  Java 普通函数

C++ 纯虚函数  ==  Java 抽象函数

C++ 抽象类    ==  Java 抽象类

C++ 虚基类    ==  Java 接口

;