>  기사  >  Java  >  Java 키워드 this(파워 노드 배열)에 대한 자세한 설명

Java 키워드 this(파워 노드 배열)에 대한 자세한 설명

黄舟
黄舟원래의
2017-03-31 10:28:271361검색

이것은 Java에서 어디에서나 볼 수 있으며 다양한 용도로 사용됩니다. 일반적인 상황에서는 이 키워드를 이해하기가 매우 쉽습니다. 그러나 처음 학습을 시작했을 때 명확하게 이해할 수 없는 질문이 있었습니다. 이제 이 글을 통해 참고할 수 있도록 기록하겠습니다. Next

Java에서 일반적으로 this 키워드를 사용할 때 이것이 이 클래스를 호출하는 메서드의 현재 인스턴스를 나타낸다는 것을 우리 모두는 알고 있습니다. 일반적인 상황에서는 이 키워드를 이해하기 쉽지만 처음 학습을 시작했을 때 명확하게 이해되지 않는 질문이 있었습니다. 이제 천천히 이해했으니 누군가도 같은 내용을 적어보고 싶습니다. 나 같은 문제. 어쩌면 다른 사람들에게 도움이 될 수도 있습니다. 먼저 일반적인 상황에서 이 역할을 간략하게 살펴보겠습니다. 예를 들어 다음 코드는

public class Leaf {
 private int i = 0;
 Leaf increment() {
  i++;
  return this;
 }
 void print() {
  System.out.println("i = " + i);
 }
 public static void main(String[] args) {
  Leaf x = new Leaf();
  x.increment().increment().print();
 }
}

Leaf 클래스의 기본 메서드에서 새 Leaf 인스턴스 x를 만든 다음 x 인스턴스가 increment() 메서드를 호출합니다. increment()가 일반적인 메소드이거나 void 메소드라면 여기서는 공부할 가치가 없습니다. 특별한 점은 increment() 메소드에서 우리가 반환하는 것은 이것이고 이것은 우리가 방금 생성한 x를 나타냅니다. x가 increment() 메서드를 호출하기 때문에 increment() 메서드는 분명히 Leaf의 x 인스턴스를 나타냅니다.

논의할 내용은 없을 것 같습니다. 이는 이 메서드를 호출하는 인스턴스 x를 나타냅니다. 그러나 main()함수를 다음과 같이 수정하면

public static void main(String[] args) {
 Leaf x = new Leaf();
 x.increment().increment().print();
  
 Leaf y = new Leaf();
 y.increment().increment().print();
}

위의 수정된 코드에서 새로운 Leaf 인스턴스 y를 생성하고 y도 두 번의 연속 호출을 호출합니다. .시간증가(). 이제 문제는 x와 y가 동시에 increment() 메서드를 호출하면 이것이 누구를 나타내는가 하는 것입니다. x가 increment() 메서드를 호출하면 이는 x를 나타냅니다. 그러나 문제는 메소드 호출에 관해 이야기할 때 JVM 레벨에서 Leaf 클래스의 increment() 메소드가 위치한 메모리 주소를 찾은 다음 Java 가상 머신 스택
에 스택 프레임을 생성한다는 것입니다.

그런 다음 스택 프레임의 메서드 내부에 있는 코드를 실행합니다. 즉, jvm 실행 메서드 수준에서는 소위 x 호출이 없으며 y가 호출됩니다. 그러면 메서드에서 이것이 가리키는 인스턴스를 어떻게 결정합니까?

Leaf 클래스 바이트코드에 어떻게 표시되는지 살펴보겠습니다. 혹시 놓친 부분이 있나요? x 인스턴스나 y 인스턴스를 메소드에 전달하지 않았다면 jvm에서 메소드를 실행하세요. 이것이 구체적으로 어떤 인스턴스를 가리키는지 아는 것은 불가능합니다.

이때 increment() 메소드에서는 인코딩에 매개변수가 없지만 바이트코드에 매개변수 개수가 1개로 표시되는 것을 알 수 있다. 결과는 이미 명확합니다. JVM이 컴파일을 수행할 때 인스턴스 메소드에서는 기본적으로 매개변수가 숨겨져 전달됩니다. 이 매개변수는 현재 호출되는 인스턴스 자체입니다. 예를 들어 x가 호출되면 숨겨진 상태에서 x가 전달되고, y가 호출되면 y가 전달됩니다. 따라서 이는 jvm 실행 메서드 수준에서 누구를 가리키는지 확인할 수 있습니다.

위의 결론은 우리만의 추론입니다. 이에 대해 자세히 설명하는 책이 있나요? "javaProgrammingThoughts"에서 이 섹션은 다음과 같이 설명됩니다.

메소드 내부에 있고 현재 객체 의 핸들을 가져오려고 한다고 가정합니다. 해당 핸들은 컴파일러에 의해 "비밀롭게" 전달되므로 사용할 수 있는 식별자가 없습니다. 그러나 이 목적을 위한 전용 키워드가 있습니다: this.

여기서 언급된 컴파일러가 비밀리에 전달한 핸들은 여기에 있는 숨겨진 매개변수입니다.

지금까지는 이에 대한 설명이 매우 명확해야 하며 JVM 수준에서 이해하고 있습니다. 그렇다면 아래 예제를 보고 기본 클래스 B가 무엇을 나타내는지 생각해 보는 데 관심이 있으신가요?

public class B {
 public B() {
  System.out.println(this.getClass().getSimpleName()); 
  System.out.println(((A) this).a); 
 }
}
public class A extends B {
 public int a = 100;  
 public A() {
  a = 200;
 } 
 public static void main(String[] args) {
  new A();
 }
}

이 예제는 원래 Java에 상속 구조가 있을 때 클래스가 초기화되는 방식을 이해하기 위한 것이었지만 클래스 B의 생성자는 매우 특별합니다. 클래스 B의 생성자에서 이에 의해 출력되는 SimpleName은 A입니다. 일반적으로 상황이 발생하면 클래스 B에서 이에 의한 SimpleName 출력은 B여야 하지만 여기서는 A입니까? 왜?

위에서 이에 대해 이야기하는 과정에서 이미 이에 대해 다루었습니다. 스택 프레임을 생성하기 위해 java 메서드를 호출할 때 jvm은 현재 인스턴스를 비밀리에 전달합니다. 따라서 A의 생성자를 실행하면 기본적으로 부모 클래스 B의 생성자가 호출됩니다. 부모 클래스 B의 생성자를 호출할 때 비밀리에 전달된 현재 인스턴스는 A의 인스턴스입니다. B가 호출되기 때문입니다. A의 생성자이므로 여기서는 대신 A를 나타냅니다.

위 내용은 Java 키워드 this(파워 노드 배열)에 대한 자세한 설명의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.