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