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的方法不能被重载,但是能够被再次声明。
方法的重载的规则:
- 方法名称必须相同。
- 参数列表必须不同(个数不同、或类型不同、参数排列顺序不同等)。
- 方法的返回类型可以相同也可以不相同。
- 仅仅返回类型不同不足以成为方法的重载。
方法重载的实现:
方法名称相同时,编译器会根据调用方法的参数个数、参数类型等去逐个匹配,以选择对应的方法,如果匹配失败,则编译器报错,这叫做重载分辨。