这几天在看《Java核心技术》这本书,看到4.3.5节-隐式参数和显式参数讲解;对this关键字的理解终于豁然开朗了!
一、显示参数、隐式参数
首先介绍一下显示参数、隐式参数
1.1 什么是显示参数
什么是显示参数,大家猜也能猜到!
显示参数就是跟在方法名后面括号里的参数!
1.2 什么是隐式参数
什么是隐式参数?说实话,这真是一个好问题,在今天之前,我对这个名词毫无概念!
隐式参数: 是把出现在方法名前类对象作为参数。(请把这句话读三遍,即调用这个方法的对象)
显式参数是明显地列在方法声明中的, 例如 double byPercent
。隐式参数没有出现在方法声明中
1.3 相关问题
1.3.1 静态方法
“静态方法”的调用总是不指定某个对象实例为“接收者”,而“实例方法”则总是要以某个对象实例为“接收者”(receiver)
静态方法属于类方法,如果在静态方法中使用this会报错!
如下:
double pow = Math.pow(2, 3);//8
在运算时,不使用任何Math对象。换句话说,没有隐式的参数。
1.3.2 构造器
这里抛出一个问题:构造器是不是静态方法?
构造器里面可以调用this,可以判断构造器不是静态方法。
- 从Java语言层面看,构造器不是静态方法。事实上规范专门规定了构造器不是方法。
- 从JVM层面看,构造器属于特殊的初始化方法,但仍然不被归类为静态方法。
- 无论从哪个层面看,构造器都是可以静态分派的(statically dispatched)。但要特别注意的是,并非所有可以静态分派的代码都是静态方法,这是规范里有定义的术语。
- 无论从哪个层面看,构造器都要接收从调用方传入的隐藏this参数。Java的静态方法不需要接收该参数,而实例方法要接收该参数。这方面构造器更接近于可以静态分派的实例方法,例如私有实例方法。这就是为什么从构造器可以调用this上的实例方法。
参考:https://rednaxelafx.iteye.com/blog/652719
二、this
this关键字和我们上述的隐式参数紧密相连!
隐式参数主要出现在实例方法和构造器中!
import java.time.LocalDate;
public class Test {
public static void main(String[] args) {
Employee employee = new Employee("xsl",20.0,2019,01,02);
Employee employee2 = null;
System.out.println(employee2);
employee2 = employee.raiseSalary(5);
System.out.println(employee2);
}
}
class Employee
{
private String name;
private double salary;
private LocalDate hireDay;
public Employee(String n, double s, int year, int month, int day)
{
this(n);//调用类的构造方法
System.out.println(this);
this.name = n;//构造器中引用变量或者成员方法
this.salary = s;
this.hireDay = LocalDate.of(year, month, day);
}
public Employee(String n) {
this.name = n;
}
public String getName()
{
return name;
}
public double getSalary()
{
return salary;
}
public LocalDate getHireDay()
{
return hireDay;
}
public Employee raiseSalary(double byPercent)
{
System.out.println(this);
double raise = this.salary * byPercent / 100;
this.salary += raise;//引用变量或者成员方法
return this;//返回对象的引用
}
}
结果:
Employee@15db9742
null
Employee@15db9742
Employee@15db9742
2.1 实例方法
在每一个方法中, 关键字 this 表示隐式参数。(构造器不是方法)
也就是说:this代表类对象的引用
“类方法”(“静态方法”)与“实例方法”在概念中的JVM上的区别:
在调用类方法时,所有参数按顺序存放于被调用方法的局部变量区中的连续区域,从局部变量0开始;
在调用实例方法时,局部变量0用于存放传入的该方法所属的对象实例(Java语言中的“this”),所有参数从局部变量1开始存放在局部变量区的连续区域中。从效果上看,这就等于在调用实例方法时总是把“this”作为第一个参数传入被调用方法。
2.1.1 在实例方法中应用
具体实现见代码
(1) 引用变量或者成员方法
(2)返回对象的引用
2.2 构造器
2.2.1在构造器中应用
(1)引用变量或者成员方法
(2)调用类的构造方法
以下构造函数、构造方法、构造器都是同义词
必须注意:就算是构造器调用构造方法,也必须为于其第一行,
由于super调用父类的构造函数也必须放在构造方法的第一行中执行,因此,通过this和super调用构造方法不能同时出现一个构造方法中。
构造方法也只能调用一个且仅一次构造器!
参考: