在整个Java中,只要是代码开发,几乎都离不开this。在Java中this可以完成三件事情:表示本类属性、表示本类方法、当前对象。
一、“this.属性”表示本类属性
讲解之前,我们先看一段代码:
package com.wz.thisdemo;
class Person {
private String name ;
private int age ;
public Person(String n,int a) {
name = n ;
age = a ;
}
// setter、getter略
public String getInfo() {
return "姓名:" + name + ",年龄:" + age ;
}
}
public class TestDemo {
public static void main(String args[]) {
Person per = new Person("张三",20) ;
System.out.println(per.getInfo()) ;
}
}
运行结果:
姓名:张三,年龄:20
我们来观察一下构造方法:
public Person(String n,int a) {
name = n ;
age = a ;
}
这个时候的构造方法的两个参数的目的是为类中的name和age两个属性初始化,可是这个方法上的两个参数,一个是字母n,另外一个是字母a,什么意思?那么最好的做法,既然构造方法的两个参数是为了name和age属性初始化使用的,最好将其的名称也定义为name和age才最为合适:
public Person(String name,int age) {
name = name ;
age = age ;
}
此时构造方法上的两个参数的名称变得有意义了。但是这样一来,一个问题就出现了,发现属性没有内容了。因为在程序之中是以“{}”作为一个分界,采用就近的取用原则,所以现在为了可以明确的指定要操作的是类中属性的话,那么应该采用“this.属性”的形式完成,代码应该变为:
public Person(String name,int age) {
this.name = name ;
this.age = age ;
}
一个开发建议:
只要是调用本类中的属性,都使用“this.属性”的方式来进行。
二、this调用本类方法
一个类中的方法分为两种:
(1)普通方法:如果现在要调用的是本类方法,可以使用“this.方法()”调用;
(2)构造方法:调用构造方法使用“this(参数…)”调用。
例如:现在一个类之中存在了三个构造方法(无参、有一个参数、有两个参数),但是不管使用何种构造方法,都要求在实例化对象产生的时候输出一行提示信息:“一个新的类对象被实例化”。按照之前的学习,代码编写如下:
package com.wz.thisdemo;
class Person {
private String name ;
private int age ;
public Person() {
System.out.println("*** 一个新的Person类对象被实例化。") ;
}
public Person(String name) {
System.out.println("*** 一个新的Person类对象被实例化。") ;
this.name = name ;
}
public Person(String name,int age) {
System.out.println("*** 一个新的Person类对象被实例化。") ;
this.name = name ;
this.age = age ;
} // setter、getter略
public String getInfo() {
return "姓名:" + this.name + ",年龄:" + this.age ;
}
}
public class TestDemo {
public static void main(String args[]) {
Person per = new Person("张三",20) ;
System.out.println(per.getInfo()) ;
}
}
运行结果:
*** 一个新的Person类对象被实例化。
姓名:张三,年龄:20
不过,遗憾的是,此时的程序之中出现了大量的重复代码,而我们的目标是尽量没有重复。这种情况下就可以利用this()来完成:
package com.wz.thisdemo;
class Person {
private String name ;
private int age ;
public Person() {
System.out.println("*** 一个新的Person类对象被实例化。") ;
}
public Person(String name) {
this() ; // 调用无参构造
this.name = name ;
}
public Person(String name,int age) {
this(name) ; // 调用有一个参数的构造
this.age = age ;
} // setter、getter略
public String getInfo() {
return "姓名:" + this.name + ",年龄:" + this.age ;
}
}
public class TestDemo {
public static void main(String args[]) {
Person per = new Person("张三",20) ;
System.out.println(per.getInfo()) ;
}
}
运行结果:
*** 一个新的Person类对象被实例化。
姓名:张三,年龄:20
以上就使用this()就完成了构造方法之间的互相调用操作。
但是要注意的是:所有的构造方法是在对象实例化的时候被默认调用,而且是在调用普通方法之前调用,所以使用“this()”调用构造方法的操作,一定要放在构造方法的首行;
public Person(String name) {
this() ; // 调用无参构造,放在首行
this.name = name ;
}
另外,我们来看一段代码:
package com.wz.thisdemo;
class Person {
private String name ;
private int age ;
public Person() {
this("",10) ; // 调用两个参数的构造
System.out.println("*** 一个新的Person类对象被实例化。") ;
}
public Person(String name) {
this() ; // 调用一个参数的构造
this.name = name ;
}
public Person(String name,int age) {
this(name) ; // 调用有一个参数的构造
this.age = age ;
} // setter、getter略
public String getInfo() {
return "姓名:" + this.name + ",年龄:" + this.age ;
}
}
public class TestDemo {
public static void main(String args[]) {
Person per = new Person("张三",20) ;
System.out.println(per.getInfo()) ;
}
}
运行结果:
Exception in thread "main" java.lang.Error: Unresolved compilation problems:
Recursive constructor invocation Person(String, int)
Recursive constructor invocation Person()
Recursive constructor invocation Person(String)
at com.wz.thisdemo.Person.<init>(TestDemo.java:6)
at com.wz.thisdemo.TestDemo.main(TestDemo.java:23)
通过以上代码和运行结果不难看出:递归调用了构造器。
所以,如果一个类之中存在了多个构造方法的话,并且这些构造方法都使用了this()互相调用,那么至少要保留一个构造方法没有调用其他构造,以作为程序的出口。
我们来看一个构造方法互相操作的实例:
定义一个雇员类(员工号、姓名、薪水、部门),在这个类中提供四个工作方法:
A、单参,只传递员工号,则员工姓名:无名氏,薪水:0,部门:未定
B、双参,传递员工号,姓名,则员工薪水为1000,部门:后勤
C、四参,传递员工号,姓名,部门,薪水
D、无参,则均为空值
来看代码:
package com.wz.thisdemo;
class Emp {
private int empno ;
private String ename ;
private double salary ;
private String dept ;
public Emp(){}
public Emp(int empno){
this.empno = empno ;
this.ename = "无名氏" ;
this.salary = 0.0 ;
this.dept = "未定" ;
}
public Emp(int empno,String ename){
this.empno = empno ;
this.ename = ename ;
this.salary = 1000.0 ;
this.dept = "后勤" ;
}
public Emp(int empno,String ename,double salary,String dept){
this.empno = empno ;
this.ename = ename ;
this.salary = salary ;
this.dept = dept ;
}
public String getInfo() {
return "雇员编号:" + this.empno + ",姓名:" + this.ename + ",工资:" + this.salary + ",部门:" + this.dept ;
}
}
public class TestDemo {
public static void main(String args[]) {
Emp emp1 = new Emp();
System.out.println(emp1.getInfo());
Emp emp2 = new Emp(4555);
System.out.println(emp2.getInfo());
Emp emp3 = new Emp(6542,"小王");
System.out.println(emp3.getInfo());
Emp emp4 = new Emp(4578,"小明",12000,"技术部");
System.out.println(emp4.getInfo());
}
}
运行结果:
雇员编号:0,姓名:null,工资:0.0,部门:null
雇员编号:4555,姓名:无名氏,工资:0.0,部门:未定
雇员编号:6542,姓名:小王,工资:1000.0,部门:后勤
雇员编号:4578,姓名:小明,工资:12000.0,部门:技术部
如果按照以上的方式开发,是可以完成功能的实现,但是代码中却存在了重复的代码,很明显,这样不合适。于是,我们通过构造方法的相互调用来对构造方法进行如下改动:
public Emp(){}
public Emp(int empno){
this(empno,"无名氏",0.0,"未定") ;//调用四参构造方法
}
public Emp(int empno,String ename){
this(empno,ename,1000.0,"后勤") ;//调用四参构造方法
}
public Emp(int empno,String ename,double salary,String dept){
this.empno = empno ;
this.ename = ename ;
this.salary = salary ;
this.dept = dept ;
}
运行结果如下:
雇员编号:0,姓名:null,工资:0.0,部门:null
雇员编号:4555,姓名:无名氏,工资:0.0,部门:未定
雇员编号:6542,姓名:小王,工资:1000.0,部门:后勤
雇员编号:4578,姓名:小明,工资:12000.0,部门:技术部
这种构造方法的互相调用是在对象实例化的时候,不同的构造有一些相同操作的情况下去使用。
三、this表示当前对象
当前对象,是指当前正在调用类中方法的对象。
先来看一段代码:
package com.wz.thisdemo;
class Book{
}
public class TestDemo {
public static void main(String args[]) {
Book booka = new Book();
System.out.println(booka);
Book bookb = new Book();
System.out.println(bookb);
}
}
运行结果:
com.wz.thisdemo.Book@15db9742
com.wz.thisdemo.Book@6d06d69c
在Book类中加上一个打印this的方法:
package com.wz.thisdemo;
class Book{
public void print(){
//this就是当前调用该方法的对象
System.out.println("this = "+ this);
}
}
public class TestDemo {
public static void main(String args[]) {
Book booka = new Book();
System.out.println(booka);
booka.print();
System.out.println();
Book bookb = new Book();
System.out.println(bookb);
bookb.print();
}
}
运行结果:
com.wz.thisdemo.Book@15db9742
this = com.wz.thisdemo.Book@15db9742
com.wz.thisdemo.Book@6d06d69c
this = com.wz.thisdemo.Book@6d06d69c
那么,之前的“this.属性”实际上就是当前对象的属性,一定是堆内存中保存的内容。