java中的this关键字到底指什么

Python018

java中的this关键字到底指什么,第1张

如果有两个同类型的对象,分别叫作a和b,那么您也许不知道如何为这两个对象同时调用一个f()方法:

class Banana { void f(int i) { /* ... */ } }

Banana a = new Banana(), b = new Banana()

a.f(1)

b.f(2)

若只有一个名叫f()的方法,它怎样才能知道自己是为a还是为b调用的呢?

为了能用简便的、面向对象的语法来书写代码——亦即“将消息发给对象”,编译器为我们完成了一些幕后工作。其中的秘密就是第一个自变量传递给方法f(),而且那个自变量是准备操作的那个对象的句柄。所以前述的两个方法调用就变成了下面这样的形式:

Banana.f(a,1)

Banana.f(b,2)

这是内部的表达形式,我们并不能这样书写表达式,并试图让编译器接受它。但是,通过它可理解幕后到底发生了什么事情。

假定我们在一个方法的内部,并希望获得当前对象的句柄。由于那个句柄是由编译器“秘密”传递的,所以没有标识符可用。然而,针对这一目的有个专用的关键字:this。this关键字(注意只能在方法内部使用)可为已调用了其方法的那个对象生成相应的句柄。可象对待其他任何对象句柄一样对待这个句柄。但要注意,假若准备从自己某个类的另一个方法内部调用一个类方法,就不必使用this。只需简单地调用那个方法即可。当前的this句柄会自动应用于其他方法。所以我们能使用下面这样的代码:

class Apricot {

void pick() { /* ... */ }

void pit() { pick()/* ... */ }

}

在pit()内部,我们可以说this.pick(),但事实上无此必要。编译器能帮我们自动完成。this关键字只能用于那些特殊的类——需明确使用当前对象的句柄。例如,假若您希望将句柄返回给当前对象,那么它经常在return语句中使用。

由于increment()通过this关键字返回当前对象的句柄,所以可以方便地对同一个对象执行多项操作。

1. 在构建器里调用构建器

若为一个类写了多个构建器,那么经常都需要在一个构建器里调用另一个构建器,以避免写重复的代码。可用this关键字做到这一点。

通常,当我们说this的时候,都是指“这个对象”或者“当前对象”。而且它本身会产生当前对象的一个句柄。在一个构建器中,若为其赋予一个自变量列表,那么this关键字会具有不同的含义:它会对与那个自变量列表相符的构建器进行明确的调用。这样一来,我们就可通过一条直接的途径来调用其他构建器。如下所示:

其中,构建器Flower(String s,int petals)向我们揭示出这样一个问题:尽管可用this调用一个构建器,但不可调用两个。除此以外,构建器调用必须是我们做的第一件事情,否则会收到编译程序的报错信息。

这个例子也向大家展示了this的另一项用途。由于自变量s的名字以及成员数据s的名字是相同的,所以会出现混淆。为解决这个问题,可用this.s来引用成员数据。经常都会在Java代码里看到这种形式的应用,本书的大量地方也采用了这种做法。

在print()中,我们发现编译器不让我们从除了一个构建器之外的其他任何方法内部调用一个构建器。

new 在创建对象时使用 例如String s=new String("abc")

java中的this随处可见,用法也多,现在整理有几点:

1. this是指当前对象自己。

当在一个类中要明确指出使用对象自己的的变量或函数时就应该加上this引用。如下面这个例子中:

public class Hello {

String s = "Hello"

public Hello(String s){

System.out.println("s = " + s)

System.out.println("1 ->this.s = " + this.s)

this.s = s

System.out.println("2 ->this.s = " + this.s)

}

public static void main(String[] args) {

Hello x=new Hello("HelloWorld!")

}

}

运行结果:

s = HelloWorld!

1 ->this.s = Hello

2 ->this.s = HelloWorld!

在这个例子中,构造函数Hello中,参数s与类Hello的变量s同名,这时如果直接对s进行操作则是对参数s进行操作。若要对类Hello的成员变量s进行操作就应该用this进行引用。运行结果的第一行就是直接对构造函数中传递过来的参数s进行打印结果; 第二行是对成员变量s的打印;第三行是先对成员变量s赋传过来的参数s值后再打印,所以结果是HelloWorld!

2. 把this作为参数传递

当你要把自己作为参数传递给别的对象时,也可以用this。如:

public class A {

public A() {

new B(this).print()

}

public void print() {

System.out.println("Hello from A!")

}

}

public 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!

在这个例子中,对象A的构造函数中,用new B(this)把对象A自己作为参数传递给了对象B的构造函数。

3. 注意匿名类和内部类中的中的this。

有时候,我们会用到一些内部类和匿名类,如事件处理。当在匿名类中用this时,这个this则指的是匿名类或内部类本身。这时如果我们要使用外部类的方法和变量的话,则应该加上外部类的类名。如下面这个例子:

public class A {

int i = 1

public A() {

Thread thread = new Thread() {

public void run() {

for() {

A.this.run()

try {

sleep(1000)

} catch(InterruptedException ie) { }

}

}

}//注意这里有

thread.start()

}

public void run() {

System.out.println("i = " + i)

i++

}

public static void main(String[] args) throws Exception {

new A()

}

}

在上面这个例子中, thread 是一个匿名类对象,在它的定义中,它的 run 函数里用到了外部类的 run 函数。这时由于函数同名,直接调用就不行了。这时有两种办法,一种就是把外部的 run 函数换一个名字,但这种办法对于一个开发到中途的应用来说是不可取的。那么就可以用这个例子中的办法用外部类的类名加上 this 引用来说明要调用的是外部类的方法 run。

4。在构造函数中,通过this可以调用同一class中别的构造函数,如

public class Flower{

Flower (int petals){}

Flower(String ss){}

Flower(int petals, Sting ss){

//petals++调用另一个构造函数的语句必须在最起始的位置

this(petals)

//this(ss)会产生错误,因为在一个构造函数中只能调用一个构造函数

}

}

值得注意的是:

1:在构造调用另一个构造函数,调用动作必须置于最起始的位置。

2:不能在构造函数以外的任何函数内调用构造函数。

3:在一个构造函数内只能调用一个构造函数。