📜  Java this关键字

📅  最后修改于: 2020-09-26 15:40:22             🧑  作者: Mango

在本文中,我们将通过示例了解Java中的此关键字,以及如何以及在何处使用它们。

这个关键字

在Java中,此关键字用于引用方法或构造函数中的当前对象。例如,

class Main {
    int instVar;

    Main(int instVar){
        this.instVar = instVar;
        System.out.println("this reference = " + this);
    }

    public static void main(String[] args) {
        Main obj = new Main(8);
        System.out.println("object reference = " + obj);
    }
}

输出

this reference = com.ThisAndThat.MyClass@74a14482
object reference = com.ThisAndThat.MyClass@74a14482

在上面的示例中,我们创建了一个Main类的名为obj的对象。然后,我们打印对对象obj和该类的this关键字的引用。

在这里,我们可以看到obj的引用与this相同。这意味着这不过是对当前对象的引用。


使用此关键字

在许多情况下,通常都使用this关键字。

使用它作为歧义变量名

在Java中,不允许在范围(类范围或方法范围)内声明两个或多个具有相同名称的变量。但是,实例变量和参数可能具有相同的名称。例如,

class MyClass {
    // instance variable
    int age;

    // parameter
    MyClass(int age){
        age = age;
    }
}

在上面的程序中,实例变量和参数具有相同的名称:age。在这里,由于名称不明确,Java编译器感到困惑。

在这种情况下,我们使用此关键字。例如,

首先,让我们看一个不使用this关键字的示例:

class Main {

    int age;
    Main(int age){
        age = age;
    }

    public static void main(String[] args) {
        Main obj = new Main(8);
        System.out.println("obj.age = " + obj.age);
    }
}

输出

mc.age = 0

在上面的示例中,我们将8作为值传递给了构造函数。但是,我们得到0作为输出。这是因为Java编译器由于实例变量和参数之间的名称不明确而感到困惑。

现在,让我们使用this关键字重写上面的代码。

class Main {

    int age;
    Main(int age){
        this.age = age;
    }

    public static void main(String[] args) {
        Main obj = new Main(8);
        System.out.println("obj.age = " + obj.age);
    }
}

输出

obj.age = 8

现在,我们正在获得预期的输出。这是因为当调用构造函数, this构造函数内部由obj对象已调用构造函数替换。因此,将年龄变量分配为值8

另外,如果参数和实例变量的名称不同,则编译器会自动附加此关键字。例如,代码:

class Main {
    int age;

    Main(int i) {
        age = i;
    }
}

等效于:

class Main {
    int age;

    Main(int i) {
        this.age = i;
    }
}

这个与Getters和Setters

this关键字的另一个常见用法是在类的setter和getters方法中。例如:

class Main {
   String name;

   // setter method
   void setName( String name ) {
       this.name = name;
   }

   // getter method
   String getName(){
       return this.name;
   }

   public static void main( String[] args ) {
       Main obj = new Main();

       // calling the setter and the getter method
       obj.setName("Toshiba");
       System.out.println("obj.name: "+obj.getName());
   }
}

输出

obj.name: Toshiba

在这里,我们使用了this关键字:

  • 在setter方法中分配值
  • 在getter方法中访问值

在构造函数重载中使用它

在处理构造函数重载时,我们可能不得不从另一个构造函数调用一个构造函数。在这种情况下,我们不能显式调用构造函数。相反,我们必须使用this关键字。

在这里,我们使用此关键字的另一种形式。即this() 。让我们举个例子

class Complex {

    private int a, b;

    // constructor with 2 parameters
    private Complex( int i, int j ){
        this.a = i;
        this.b = j;
    }

    // constructor with single parameter
    private Complex(int i){
        // invokes the constructor with 2 parameters
        this(i, i); 
    }

    // constructor with no parameter
    private Complex(){
        // invokes the constructor with single parameter
        this(0);
    }

    @Override
    public String toString(){
        return this.a + " + " + this.b + "i";
    }

    public static void main( String[] args ) {
  
        // creating object of Complex class
        // calls the constructor with 2 parameters
        Complex c1 = new Complex(2, 3); 
    
        // calls the constructor with a single parameter
        Complex c2 = new Complex(3);

        // calls the constructor with no parameters
        Complex c3 = new Complex();

        // print objects
        System.out.println(c1);
        System.out.println(c2);
        System.out.println(c3);
    }
}

输出

2 + 3i
3 + 3i
0 + 0i

在上面的示例中,我们使用了this关键字,

  • 从构造函数Complex(int i)调用构造函数Complex(int i, int j) Complex(int i)
  • 从构造函数Complex()调用构造函数Complex(int i) Complex()

注意这一行,

System.out.println(c1);

在这里,当我们打印对象c1时 ,该对象将转换为字符串。在此过程中,将调用toString() 。由于我们在类内部重写了toString()方法,因此我们根据该方法获得输出。

this()的巨大优势之一是减少重复代码的数量。但是,在使用this()时我们应始终小心。

这是因为从另一个构造函数调用构造函数会增加开销,并且这是一个缓慢的过程。使用this()另一个巨大优势是减少重复代码的数量。

注意 :从另一个构造函数调用一个构造函数称为显式构造函数调用。


作为参数传递

我们可以使用this关键字将当前对象作为参数传递给方法。例如,

class ThisExample {
    // declare variables
    int x;
    int y;

    ThisExample(int x, int y) {
       // assign values of variables inside constructor
        this.x = x;
        this.y = y;

        // value of x and y before calling add()
        System.out.println("Before passing this to addTwo() method:");
        System.out.println("x = " + this.x + ", y = " + this.y);

        // call the add() method passing this as argument
        add(this);

        // value of x and y after calling add()
        System.out.println("After passing this to addTwo() method:");
        System.out.println("x = " + this.x + ", y = " + this.y);
    }

    void add(ThisExample o){
        o.x += 2;
        o.y += 2;
    }
}

class Main {
    public static void main( String[] args ) {
        ThisExample obj = new ThisExample(1, -2);
    }
}

输出

Before passing this to addTwo() method:
x = 1, y = -2
After passing this to addTwo() method:
x = 3, y = 0

在上面的示例中,在构造函数ThisExample() ,请注意以下行:

add(this);

在这里,我们通过将add()作为参数传递来调用add()方法。由于此关键字包含对类的对象obj的引用,因此我们可以在add()方法内更改xy的值。