前言

这次我们来回顾一下thissuper这两个关键字的用法,作为一名Java程序员,我觉得基础是最重要的,因为它决定了我们的上限,所以我的文章大部分还是以分享Java基础知识为主,学好基础,后面的知识我想学起来就变得简单。废话不多说,进入正文。

this

this 关键字只能在方法内部使用,表示对调用方法的那个对象的引用。

其实简单来说 this 关键字就是表示当前对象,下面我们来具体介绍 this 关键字在Java中的用法。

1、调用成员变量

在一个类的方法内部,如果我们想调用其成员变量,不用 this,我们会怎么做?

public class ThisTest {

    private String name = "xiaoming";

    public String getName() {
return name;
} public void setName(String name) {
name = name;
}
}

看上面的代码,我们在 ThisTest 类中创建了一个 name 属性,然后创建了一个 setName 方法,注意这个方法的形参也是 String name,那么我们通过 name = name 这样赋值,会改变成员变量 name 的属性吗?

public static void main(String[] args) {
ThisTest thisTest = new ThisTest();
thisTest.setName("xiaoma");
System.out.println(thisTest.getName());
}

打印结果是 xiaoming,而不是我们重新设置的 xiaoma,显然这种方式是不能在方法内部调用到成员变量的。因为形参的名字和成员变量的名字相同,setName 方法内部的 name = name,根据最近原则,编译器默认是将这两个 name 属性都解析为形参 name,从而导致我们设值操作和成员变量 name 完全没有关系,当然设置不了。

解决办法就是使用 this 关键字。我们将 setName 方法修改如下:

public void setName(String name) {
this.name = name;
}

在调用上面的 main 方法进行赋值,打印的结果就是 xiaoma了。

this 表示当前对象,也就是调用该方法的对象,对象.name 肯定就是调用的成员变量。

2、调用构造方法

构造方法是与类同名的一个方法,构造方法没有返回值,但是也不能用 void 来修饰。在一个类中,必须存在一个构造方法,如果没有,编译器会在编译的时候自动为这个类添加一个无参构造方法。一个类能够存在多个构造方法,调用的时候根据参数来区分。

public class Student {

    private int age;

    private String name;

    public Student() {
this("小马",50);
} public Student(String name, int age) {
this.name = name;
this.age = age;
System.out.println(name + "今年" + age + "岁了");
} public static void main(String[] args) {
Student student01 = new Student();
Student student02 = new Student("小军",45);
}
}

通过this("小马",50)来调用另外一个构造方法 Student(String name, int age) 来给成员变量初始化赋值。

输出结果:

小马今年50岁了
小军今年45岁了 Process finished with exit code 0

注意:通过 this 来调用构造方法,只能将这条代码放在构造函数的第一行,这是编译器的规定,如下所示:放在第二行会报错。

3、调用普通方法

this 表示当前对象,那么肯定能够调用当前类的普通方法。

public Student() {
this.say();
} public void say(){
System.out.println("小马很会唱歌。");
}

4、返回当前对象

public class ThisTest {

    public Object newObject(){
return this;
}
}

这表示的意思是谁调用 newObject() 方法,那么就返回谁的引用。

super

Java 中的 super 关键字则是表示 父类对象的引用

我们分析这句话父类对象的引用,那说明我们使用的时候只能在子类中使用,既然是对象的引用,那么我们也可以用来调用成员属性以及成员方法,当然了,这里的 super 关键字还能够调用父类的构造方法。

具体有如下几种用法

1、调用父类的构造方法

Java中的继承大家都应该了解,子类继承父类,我们是能够用子类的对象调用父类的属性和方法的,我们知道属性和方法只能够通过对象调用,那么我们可以大胆假设一下:在创建子类对象的同时,也创建了父类的对象,而创建对象是通过调用构造函数实现的,那么我们在创建子类对象的时候,应该会调用父类的构造方法。

下面我们看这段代码:

public class Teacher {

   public Teacher(){
System.out.println("我是一名人民教师。");
}
} class Student extends Teacher { public Student(){
System.out.println("我是一名学生。");
}
}

下面我们创建子类的对象:

public static void main(String[] args) {
Student s = new Student();
}

输出结果:

我是一名人民教师。
我是一名学生。 Process finished with exit code 0

通过打印结果看到我们在创建子类对象的时候,首先调用了父类的构造方法,接着调用子类的构造方法,也就是说在创建子类对象的时候,首先创建了父类对象,与前面我们猜想的一致。

那么问题又来了:是在什么时候调用的父类构造方法呢?

可以参考Java官方文档:https://docs.oracle.com/javase/specs/jls/se8/html/jls-8.html#d5e14278

红色框内的英文翻译为:如果声明的类是原始类Object,那么默认的构造函数有一个空的主体。否则,默认构造函数只是简单地调用没有参数的超类构造函数。

也就是说:除了顶级类 Object.class 构造函数没有调用父类的构造方法,其余的所有类都默认在构造函数中调用了父类的构造函数(没有显式声明父类的子类其父类是 Object)。

那么是通过什么来调用的呢?我们接着看官方文档:

上面的意思大概就是:超类构造函数通过 super 关键字调用,并且是以 super 关键字开头。

所以上面的 Student类的构造方法实际上应该是这样的:

class Student extends Teacher {

    public Student(){
super();//子类通过super调用父类的构造方法
System.out.println("我是一名学生。");
} public static void main(String[] args) {
Student s = new Student();
}
}

子类默认是通过 super() 调用父类的无参构造方法,如果父类显示声明了一个有参构造方法,而没有声明无参构造方法,实例化子类是会报错的。

 解决办法就是通过 super 关键字调用父类的有参构造方法:

class Student extends Teacher {

    public Student(){
super("小马");
System.out.println("我是一名学生。");
} public static void main(String[] args) {
Student s = new Student();
}
}

2、调用父类的成员属性

public class Teacher {

    public String name = "小马";

    public Teacher() {
System.out.println("我是一名人民教师。");
}
} class Student extends Teacher { public Student() {
System.out.println("我是一名学生。");
} public void fatherName() {
System.out.println("我的父类名字是:" + super.name);//调用父类的属性
} public static void main(String[] args) {
Student student = new Student();
student.fatherName();
}
}

输出结果:

我是一名人民教师。
我是一名学生。
我的父类名字是:小马 Process finished with exit code 0

3、调用父类的方法

public class Teacher {

    public String name;

    public Teacher() {
System.out.println("我是一名人民教师。");
} public void setName(String name){
this.name = name;
}
} class Student extends Teacher { public Student() {
super();//调用父类的构造方法
System.out.println("我是一名学生。");
} public void fatherName() {
super.setName("小军");//调用父类普通方法
System.out.println("我的父类名字是:" + super.name);//调用父类的属性
} public static void main(String[] args) {
Student student = new Student();
student.fatherName();
}
}

输出结果:

我是一名人民教师。
我是一名学生。
我的父类名字是:小军 Process finished with exit code 0

4、this 和 super 出现在同一个构造方法中?

假设 super()this() 关键字的前面

首先通过 super() 调用父类构造方法,对父类进行一次实例化。接着调用 this()this() 方法会调用子类的构造方法,在子类的构造方法中又会对父类进行一次实例化。也就是说我们对子类进行一次实例化,对造成对父类进行两次实例化,所以显然编译器是不允许的。

反过来 this()super() 之前也是一样。而且编译器有限定 this()super() 这两个关键字都只能出现在构造方法的第一行,将这两个关键字放在一起,总有一个关键字在第二行,编译是不能通过的。

this和super异同

  • super(参数)调用基类中的某一个构造函数(应该为构造函数中的第一条语句)。
  • this(参数)调用本类中另一种形成的构造函数(应该为构造函数中的第一条语句)。
  • super: 它引用当前对象的直接父类中的成员(用来访问直接父类中被隐藏的父类中成员数据或函数,基类与派生类中有相同成员定义时如:super.变量名super.成员函数据名(实参)
  • this:它代表当前对象名(在程序中易产生二义性之处,应使用 this 来指明当前对象;如果函数的形参与类中的成员数据同名,这时需用 this 来指明成员变量名)。
  • 调用super()必须写在子类构造方法的第一行,否则编译不通过。每个子类构造方法的第一条语句,都是隐含地调用 super(),如果父类没有这种形式的构造函数,那么在编译的时候就会报错。
  • super()this() 类似,区别是,super() 从子类中调用父类的构造方法,this() 在同一类内调用其它方法。
  • super()this() 均需放在构造方法内第一行。
  • 尽管可以用this调用一个构造器,但却不能调用两个。
  • thissuper 不能同时出现在一个构造函数里面,因为this必然会调用其它的构造函数,其它的构造函数必然也会有 super 语句的存在,所以在同一个构造函数里面有相同的语句,就失去了语句的意义,编译器也不会通过。
  • this()super() 都指的是对象,所以,均不可以在 static 环境中使用。包括:static 变量,static 方法,static 语句块
  • 从本质上讲,this 是一个指向本对象的指针, 然而 super 是一个 Java 关键字。

结尾

我是一个正在被打击还在努力前进的码农。如果文章对你有帮助,记得点赞、关注哟,谢谢!

Java 中 this 和 super 的用法详解的更多相关文章

  1. Java中static变量作用和用法详解

    static表示"全局"或者"静态"的意思,用来修饰成员变量和成员方法,也可以形成静态static代码块,但是Java语言中没有全局变量的概念. 被static ...

  2. Java中链接MS SQL 数据库用法详解

    一.第一种方法: 使用JDBC-ODBC的桥方式 JDBC-ODBC桥连接器是用JdbcOdbc.class 和一个用于访问ODBC驱动程序的本地库实现的,对于Windows平台,该本地库是一个动态链 ...

  3. Java 5 的新标准语法和用法详解集锦

    Java 5 的新标准语法和用法详解集锦 Java 5 的新标准语法和用法详解集锦 (需要在首选项-java-complier-compiler compliance level中设置为java5.0 ...

  4. java中List的用法和实例详解

    java中List的用法和实例详解 List的用法List包括List接口以及List接口的所有实现类.因为List接口实现了Collection接口,所以List接口拥有Collection接口提供 ...

  5. Java中堆内存和栈内存详解2

    Java中堆内存和栈内存详解   Java把内存分成两种,一种叫做栈内存,一种叫做堆内存 在函数中定义的一些基本类型的变量和对象的引用变量都是在函数的栈内存中分配.当在一段代码块中定义一个变量时,ja ...

  6. 教程-Delphi中Spcomm使用属性及用法详解

    Delphi中Spcomm使用属性及用法详解 Delphi是一种具有 功能强大.简便易用和代码执行速度快等优点的可视化快速应用开发工具,它在构架企业信息系统方面发挥着越来越重要的作用,许多程序员愿意选 ...

  7. Java中的equals和hashCode方法详解

    Java中的equals和hashCode方法详解  转自 https://www.cnblogs.com/crazylqy/category/655181.html 参考:http://blog.c ...

  8. 转:Java中的equals和hashCode方法详解

    转自:Java中的equals和hashCode方法详解 Java中的equals方法和hashCode方法是Object中的,所以每个对象都是有这两个方法的,有时候我们需要实现特定需求,可能要重写这 ...

  9. js数组中foEach和map的用法详解 jq中的$.each和$.map

    数组中foEach和map的用法详解 相同点: 1.都是循环遍历数组(仅仅是数组)中的每一项. 2.forEach() 和 map() 里面每一次执行匿名函数都支持3个参数:数组中的当前项value, ...

随机推荐

  1. Java JDK 动态代理(AOP)使用及实现原理分析

    一.什么是代理? 代理是一种常用的设计模式,其目的就是为其他对象提供一个代理以控制对某个对象的访问.代理类负责为委托类预处理消息,过滤消息并转发消息,以及进行消息被委托类执行后的后续处理. 代理模式U ...

  2. 【题解】Luogu P2327 [SCOI2005]扫雷

    Luogu P2327 [SCOI2005]扫雷 Description 相信大家都玩过扫雷的游戏.那是在一个n*m的矩阵里面有一些雷,要你根据一些信息找出雷来.万圣节到了,"余" ...

  3. 一次性搞清Java中的类加载问题

    摘要:很多时候提到类加载,大家总是没法马上回忆起顺序,这篇文章会用一个例子为你把类加载的诸多问题一次性澄清. 本文分享自华为云社区<用1个例子加5个问题,一次性搞清java中的类加载问题[奔跑吧 ...

  4. 免费获取思维导图Mindmaster会员教程

    免费获取思维导图Mindmaster会员教程 步骤一.下载安装 搜索亿图官方网页,下载安装免费版Mindmaster思维导图. 步骤二.点击登录 打开Mindmaster思维导图,点击登录,可以使用微 ...

  5. 单臂路由&链路捆绑

    单臂路由&链路捆绑 目录 一.单臂路由 1.1.单臂路由作用 1.2.单臂路由实现不同VLAN间通信的原理 二.单臂路由相关命令配置 三.链路捆绑 3.1.以太网链路聚合原理 3.2.命令配置 ...

  6. react中 props,state,render函数的关系

    1.当组件的 state 或者 props 发生改变的时候,自己的render函数就会重新执行. 2. 当父组件的render函数执行时,其所有子组件的render函数都会重新执行.

  7. Redis6使用指导(完整版)

    一.Nosql与Redis概述 二.Redis6安装 三.常用五大数据类型 四.Redis6配置文件详解 五.Redis6的发布和订阅 六.Redis6新数据类型 七.Jedis操作Redis6(Ma ...

  8. Redis的Pipeline、事务和lua

    1. Pipeline 1.1 Pipeline概念 Redis客户端执行一条命令分别为如下4个过程: 1) 发送命令 2) 命令排队 3) 命令执行 4) 返回结果 其中1)+4)称为Round T ...

  9. 单片机引脚扩展芯片74HC595手工分解实验

    我们先来看下效果 74HC595是常用的串转并芯片,支持芯片级联实现少量IO口控制多个IO口输出功能 14脚:DS,串行数据输入引脚 13脚:OE, 输出使能控制脚,它是低电才使能输出,所以接GND ...

  10. vue v-if条件判断

    <view class='circle-G' v-if="item.status === 'G'"></view> <view class='circ ...