Java this关键字详解
this 关键字用来表示当前对象本身,或当前类的一个实例,通过 this 可以调用本对象的所有方法和属性。例如:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
public class Demo{
public int x = 10 ;
public int y = 15 ;
public void sum(){
// 通过 this 点取成员变量
int z = this .x + this .y;
System.out.println( "x + y = " + z);
}
public static void main(String[] args) {
Demo obj = new Demo();
obj.sum();
}
}
|
运行结果:
1
|
x + y = 25
|
上面的程序中,obj 是 Demo 类的一个实例,this 与 obj 等价,执行 int z = this.x + this.y;,就相当于执行 int z = obj.x + obj.y;。
注意:this 只有在类实例化后才有意义。
使用this区分同名变量
成员变量与方法内部的变量重名时,希望在方法内部调用成员变量,怎么办呢?这时候只能使用this,例如:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
public class Demo{
public String name;
public int age;
public Demo(String name, int age){
this .name = name;
this .age = age;
}
public void say(){
System.out.println( "网站的名字是" + name + ",已经成立了" + age + "年" );
}
public static void main(String[] args) {
Demo obj = new Demo( "微学苑" , 3 );
obj.say();
}
}
|
运行结果:
1
|
网站的名字是微学苑,已经成立了3年
|
形参的作用域是整个方法体,是局部变量。在Demo()中,形参和成员变量重名,如果不使用this,访问到的就是局部变量name和age,而不是成员变量。在 say() 中,我们没有使用 this,因为成员变量的作用域是整个实例,当然也可以加上 this:
1
2
3
|
public void say(){
System.out.println( "网站的名字是" + this .name + ",已经成立了" + this .age + "年" );
}
|
Java 默认将所有成员变量和成员方法与 this 关联在一起,因此使用 this 在某些情况下是多余的。
作为方法名来初始化对象
也就是相当于调用本类的其它构造方法,它必须作为构造方法的第一句。示例如下:
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 String name;
public int age;
public Demo(){
this ( "微学苑" , 3 );
}
public Demo(String name, int age){
this .name = name;
this .age = age;
}
public void say(){
System.out.println( "网站的名字是" + name + ",已经成立了" + age + "年" );
}
public static void main(String[] args) {
Demo obj = new Demo();
obj.say();
}
}
|
运行结果:
网站的名字是微学苑,已经成立了3年
值得注意的是:
在构造方法中调用另一个构造方法,调用动作必须置于最起始的位置。
不能在构造方法以外的任何方法内调用构造方法。
在一个构造方法内只能调用一个构造方法。
上述代码涉及到方法重载,即Java允许出现多个同名方法,只要参数不同就可以。后续章节会讲解。
作为参数传递
需要在某些完全分离的类中调用一个方法,并将当前对象的一个引用作为参数传递时。例如:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
|
public class Demo{
public static void main(String[] args){
B b = new B( new A());
}
}
class A{
public A(){
new B( this ).print(); // 匿名对象
}
public void print(){
System.out.println( "Hello from A!" );
}
}
class B{
A a;
public B(A a){
this .a = a;
}
public void print() {
a.print();
System.out.println( "Hello from B!" );
}
}
|
运行结果:
1
2
|
Hello from A!
Hello from B!
|
匿名对象就是没有名字的对象。如果对象只使用一次,就可以作为匿名对象,代码中 new B(this).print(); 等价于 ( new B(this) ).print();,先通过 new B(this) 创建一个没有名字的对象,再调用它的方法。
Java方法重载
在Java中,同一个类中的多个方法可以有相同的名字,只要它们的参数列表不同就可以,这被称为方法重载(method overloading)。
参数列表又叫参数签名,包括参数的类型、参数的个数和参数的顺序,只要有一个不同就叫做参数列表不同。
重载是面向对象的一个基本特性。
下面看一个详细的实例。
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
|
public class Demo{
// 一个普通的方法,不带参数
void test(){
System.out.println( "No parameters" );
}
// 重载上面的方法,并且带了一个整型参数
void test( int a){
System.out.println( "a: " + a);
}
// 重载上面的方法,并且带了两个参数
void test( int a, int b){
System.out.println( "a and b: " + a + " " + b);
}
// 重载上面的方法,并且带了一个双精度参数
double test( double a){
System.out.println( "double a: " + a);
return a*a;
}
public static void main(String args[]){
Demo obj= new Demo();
obj.test();
obj.test( 2 );
obj.test( 2 , 3 );
obj.test( 2.0 );
}
}
|
运行结果:
1
2
3
4
|
No parameters
a: 2
a and b: 2 3
double a: 2.0
|
通过上面的实例,读者可以看出,重载就是在一个类中,有相同的函数名称,但形参不同的函数。重载的结果,可以让一个程序段尽量减少代码和方法的种类。
说明:
- 参数列表不同包括:个数不同、类型不同和顺序不同。
- 仅仅参数变量名称不同是不可以的。
- 跟成员方法一样,构造方法也可以重载。
- 声明为final的方法不能被重载。
- 声明为static的方法不能被重载,但是能够被再次声明。
方法的重载的规则:
- 方法名称必须相同。
- 参数列表必须不同(个数不同、或类型不同、参数排列顺序不同等)。
- 方法的返回类型可以相同也可以不相同。
- 仅仅返回类型不同不足以成为方法的重载。
方法重载的实现:
方法名称相同时,编译器会根据调用方法的参数个数、参数类型等去逐个匹配,以选择对应的方法,如果匹配失败,则编译器报错,这叫做重载分辨。