Java的super关键字与instanceof运算符使用方法

时间:2021-10-08 23:49:00

Java super关键字
super 关键字与 this 类似,this 用来表示当前类的实例,super 用来表示父类。

super 可以用在子类中,通过点号(.)来获取父类的成员变量和方法。super 也可以用在子类的子类中,Java 能自动向上层类追溯。

父类行为被调用,就好象该行为是本类的行为一样,而且调用行为不必发生在父类中,它能自动向上层类追溯。

super 关键字的功能:
调用父类中声明为 private 的变量。
点取已经覆盖了的方法。
作为方法名表示父类构造方法。
调用隐藏变量和被覆盖的方法

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
public class Demo{
  public static void main(String[] args) {
    Dog obj = new Dog();
    obj.move();
  }
}
class Animal{
  private String desc = "Animals are human's good friends";
  // 必须要声明一个 getter 方法
  public String getDesc() { return desc; }
  public void move(){
    System.out.println("Animals can move");
  }
}
class Dog extends Animal{
  public void move(){
    super.move(); // 调用父类的方法
    System.out.println("Dogs can walk and run");
    // 通过 getter 方法调用父类隐藏变量
    System.out.println("Please remember: " + super.getDesc());
  }
}

运行结果:

?
1
2
3
Animals can move
Dogs can walk and run
Please remember: Animals are human's good friends

move() 方法也可以定义在某些祖先类中,比如父类的父类,Java 具有追溯性,会一直向上找,直到找到该方法为止。

通过 super 调用父类的隐藏变量,必须要在父类中声明 getter 方法,因为声明为 private 的数据成员对子类是不可见的。
调用父类的构造方法

在许多情况下,使用默认构造方法来对父类对象进行初始化。当然也可以使用 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 Demo{
  public static void main(String[] args) {
    Dog obj = new Dog("花花", 3);
    obj.say();
  }
}
class Animal{
  String name;
  public Animal(String name){
    this.name = name;
  }
}
class Dog extends Animal{
  int age;
  public Dog(String name, int age){
    super(name);
    this.age = age;
  }
  public void say(){
    System.out.println("我是一只可爱的小狗,我的名字叫" + name + ",我" + age + "岁了");
  }
}

运行结果:
我是一只可爱的小狗,我的名字叫花花,我3岁了

注意:无论是 super() 还是 this(),都必须放在构造方法的第一行。

值得注意的是:
在构造方法中调用另一个构造方法,调用动作必须置于最起始的位置。
不能在构造方法以外的任何方法内调用构造方法。
在一个构造方法内只能调用一个构造方法。

如果编写一个构造方法,既没有调用 super() 也没有调用 this(),编译器会自动插入一个调用到父类构造方法中,而且不带参数。

最后注意 super 与 this 的区别:super 不是一个对象的引用,不能将 super 赋值给另一个对象变量,它只是一个指示编译器调用父类方法的特殊关键字。

Java instanceof 运算符
多态性带来了一个问题,就是如何判断一个变量所实际引用的对象的类型 。 C++使用runtime-type information(RTTI),Java 使用 instanceof 操作符。

instanceof 运算符用来判断一个变量所引用的对象的实际类型,注意是它引用的对象的类型,不是变量的类型。请看下面的代码:

?
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 final class Demo{
  public static void main(String[] args) {
    // 引用 People 类的实例
    People obj = new People();
    if(obj instanceof Object){
      System.out.println("我是一个对象");
    }
    if(obj instanceof People){
      System.out.println("我是人类");
    }
    if(obj instanceof Teacher){
      System.out.println("我是一名教师");
    }
    if(obj instanceof President){
      System.out.println("我是校长");
    }
    System.out.println("-----------"); // 分界线
    
    // 引用 Teacher 类的实例
    obj = new Teacher();
    if(obj instanceof Object){
      System.out.println("我是一个对象");
    }
    if(obj instanceof People){
      System.out.println("我是人类");
    }
    if(obj instanceof Teacher){
      System.out.println("我是一名教师");
    }
    if(obj instanceof President){
      System.out.println("我是校长");
    }
  }
}
class People{ }
class Teacher extends People{ }
class President extends Teacher{ }

运行结果:

?
1
2
3
4
5
6
我是一个对象
我是人类
-----------
我是一个对象
我是人类
我是一名教师

可以看出,如果变量引用的是当前类或它的子类的实例,instanceof 返回 true,否则返回 false。