this与super本质区别

606 阅读4分钟

「这是我参与11月更文挑战的第6天,活动详情查看:2021最后一次更文挑战

1、简介

this与super是类实例化时通往Object类通道的打通者;this和super在程序中由于其经常被隐式的使用而被我们忽略,但是理解其作用和使用规范肯定是必须的。接下来将详述this与super的作用和区别。

2、引子

先来看两段代码,无报错代码示例:

package com.liziba.ts;

/**
 * <p>
 * 		父类示例代码
 * </p>
 *
 * @Author: Liziba
 */
public class Father {

    private String name;

    public String getName() {
        return name;
    }

}
package com.liziba.ts;

/**
 * <p>
 * 		子类示例代码
 * </p>
 *
 * @Author: Liziba
 */
public class Son extends Father{

    public Son(String name) {

    }
   
}

此时将Father类中的构造函数修改为有参的构造函数,有错误的代码示例:

package com.liziba.ts;

/**
 * <p>
 * 		父类示例代码
 * </p>
 *
 * @Author: Liziba
 */
public class Father {

    private String name;

    public Father(String name) {
        this.name = name;
    }

    public String getName() {
        return name;
    }

}

子类代码不修改,此时子类报错:

这就是super()的隐式使用导致的报错,具体原因是因为子类Son的构造函数中隐式的调用了父类的无参构造器,相当于隐式的super(),上面报错的代码和下面这个是等价的。

但是由于父类没有显示的申明无参构造函数,此时无参构造函数被有参构造函数覆盖了,所有super()调用无法抵达父类。此时的解决办法有两种:

1、父类中声明无参构造函数

public class Father {

    public String name;
	
    // 父类什么无参构造函数
    public Father() {
    }

    public Father(String name) {
        this.name = name;
    }
}

public class Son extends Father {

    public Son(String name) {
        super(); // 可以省略
    }
}

2、子类显示的通过super调用父类的有参构造函数

public class Son extends Father {

    public Son(String name) {
        // 调用父类的有参构造函数 
        super(name);
    }
}

接下来将详细分析this和super的作用和区别。

3、this

this相当于当前对象实例,或者当前对象的一个引用,this有如下作用:

  1. 调用当前对象中的方法和属性
  2. 区分对象属性和方法形参
  1. 调用构造方法(必须在构造函数的第一行)

this相当于当前对象实例举例:

public class Son extends Father{

    private String homework = "Java编程思想";

    public void doSomething() {
        // this 相当于当前Son对象实例
        synchronized (this) {

        }
    }
}

调用当前对象中的方法和属性举例:

public class Son extends Father{

    private String homework = "Java编程思想";

    public void doSomething() {
        // this 获取当前对象的属性
        String hn = this.homework;
        // this 调用当前对象的方法
        this.doSomething2();
    }

    public void doSomething2() {
        // toDo
    }
}

区分对象属性和方法形参举例:

public class Son extends Father{

    private String homework = "Java编程思想";

    public Son(String homework) {
        // 区分对象属性与方法形参
        this.homework = homework;
    }
    
}

调用其他构造方法举例:

public class Son extends Father{

    private String homework = "Java编程思想";

    public Son(String homework) {
        // 调用其他构造方法,必须在第一行
        this(homework, "你们未来都是架构师");
    }

    public Son(String homework, String name) {
    }
}

4、super

super可以理解为父类(直接父类,如果有多层继承关系这里指的是最近的父类)对象的引用。super有如下的作用:

  1. 调用父类非私有的属性和方法
  2. 区分当前类与父类同名的属性和方法
  1. 调用父类的构造函数(必须在构造函数的第一行)

调用父类属性和方法示例:

/**
 *	父类
 */
public class Father {

    public String name;

    public void doSomething3() {
        // toDo
    }

    public void doSomething4() {
        // toDo
    }
}

/**
 *	子类
 */
public class Son extends Father{

   public void doSomething() {
       // 调用父类的非私有方法
       super.doSomething3();
       super.doSomething4();
	   // 调用父类的非私有属性
       String name = super.name;
   }

}

区分当前类与父类同名的属性和方法示例:

/**
 *	父类
 */
public class Father {

    public String name;

    public void doSomething3() {
        // toDo
    }

    public void doSomething4() {
        // toDo
    }
}


/**
 *	子类代码修改成如下
 */
public class Son extends Father {
    public String name;

    public void doSomething() {
		
        // super可以区分父类方法与当前对象的方法
        doSomething3();
        doSomething4();
        super.doSomething3();
        super.doSomething4();

        // 区分当前父类的属性与当前类的属性
        String fatherName = super.name;
        String sonName = name;
    }

    @Override
    public void doSomething3() {
        // todo
    }

    @Override
    public void doSomething4() {
        // todo
    }
}

调用父类的构造函数(必须在构造函数的第一行)

/**
 *	父类
 */
public class Father {

    public String name;

    public Father(String name) {
        this.name = name;
    }
}

/**
 *	子类
 */
public class Son extends Father {

    public Son(String name) {
        super(name);
    }
}

5、总结

5.1 对比差异

this 基本概念

  • 访问本类实例属性和方法

super 基本概念

  • 访问父类实例属性和方法

this 查找范围

  • 先查找本类,不存在再查找父类

super 查找范围

  • 直接查找父类

this 其他功能

  • 单独使用表示当前对象

super 其他功能

  • 子类复写父类方法,用于访问父类同名方法

5.2 相同点

  • 都是关键字,起指代作用
  • 构造方法中必须在第一行调用其它构造函数

5.3 总结图