메서드 호출(call by)은 컴퓨터 과학의 표준 용어입니다. 메소드 호출은 전달된 매개변수에 따라 값 호출(참조에 의한 호출)과 참조 호출(값에 의한 호출)로 구분됩니다. 세상에는 이 두 가지 유형의 호출에 대한 정의가 많이 있습니다. 가장 일반적인 말은 값을 전달하는 것이 값 호출이고, 주소를 전달하는 것이 참조 호출이라는 것입니다. 이는 실제로 매우 부적절합니다. 이러한 설명은 Java의 객체 매개변수 전달이 참조 호출임을 쉽게 상기시켜 줍니다. 실제로 Java의 객체 매개변수 전달은 여전히 값 호출입니다.
먼저 코드를 사용하여 Java의 객체 매개변수 전달이 값으로 호출되는 이유를 확인해 보겠습니다.
public class Employee { public String name=null; public Employee(String n){ this.name=n; } //将两个Employee对象交换 public static void swap(Employee e1,Employee e2){ Employee temp=e1; e1=e2; e2=temp; System.out.println(e1.name+" "+e2.name); //打印结果:李四 张三 } //主函数 public static void main(String[] args) { Employee worker=new Employee("张三"); Employee manager=new Employee("李四"); swap(worker,manager); System.out.println(worker.name+" "+manager.name); //打印结果仍然是: 张三 李四 } }
위 결과는 매우 실망스럽습니다. 형식 매개변수 객체 e1과 e2의 내용은 교환되지만 실제 매개변수 객체인 작업자와 관리자는 교환되지 않습니다. 콘텐츠. 여기서 가장 중요한 이유는 형식 매개변수 e1과 e2가 실제 매개변수 작업자와 관리자의 주소 복사본이기 때문입니다.
우리 모두 알고 있듯이 Java의 객체 변수 이름은 실제로 힙에 있는 객체의 주소를 나타냅니다(기술 용어로 객체 참조라고 함). Java 메소드가 호출될 때 전달되는 매개변수는 객체에 대한 참조입니다. 중요한 것은 형식 매개변수와 실제 매개변수가 차지하는 메모리 주소가 동일하지 않다는 것입니다. 형식 매개변수의 내용은 실제 매개변수에 저장된 객체 참조의 복사본일 뿐입니다.
JVM 메모리 관리에서 자바 스택의 로컬 변수 영역("Java Virtual Machine Architecture" 참조)에 대해 조금이라도 알고 있다면 위 문장을 쉽게 이해할 수 있을 것이다. JVM이 위 프로그램을 실행할 때 기본 메소드와 스왑 메소드를 실행하면 스택 프레임이라는 두 개의 메모리 공간이 Java 스택에 푸시됩니다. 실제 매개변수 개체 작업자 및 관리자에 대한 참조를 저장하기 위해 메인 스택 프레임에 로컬 변수 영역이라는 메모리 조각이 있습니다. 스왑 스택 프레임의 지역 변수 영역은 형식 매개변수 개체 e1 및 e2에 대한 참조를 저장합니다. e1과 e2의 기준값은 각각 작업자와 관리자의 기준값과 동일하지만, 서로 다른 메모리 공간을 차지한다. e1과 e2의 참조값을 교환할 경우, 작업자와 관리자의 참조값에는 전혀 영향을 미치지 않음을 아래 그림을 통해 확실히 알 수 있습니다.
Java 객체 매개변수 전달이 주소(참조)를 전달하더라도 여전히 값 호출입니다. 이제 참조별 호출과 값별 호출에 대한 정확한 정의를 내릴 차례입니다.
값에 의한 호출: 매개변수 전달 과정에서 형식 매개변수와 실제 매개변수는 완전히 다른 두 개의 메모리 공간을 차지합니다. 형식 매개변수에 저장된 내용은 실제 매개변수에 저장된 내용의 복사본입니다. 실제로 Java 객체의 전송은 형식 매개변수와 실제 매개변수에 저장된 내용이 일반적인 의미의 변수 값이 아니라 변수의 주소라는 점을 제외하면 이 정의를 따릅니다. 기침하고 다시 생각해보세요. 변수의 주소도 값이 아닌가요?
참조에 의한 호출: 매개변수 전송 과정에서 형식 매개변수와 실제 매개변수는 완전히 동일한 메모리 공간에 있으며 서로 분리되지 않습니다. 실제로 프로그래밍에서는 형식적인 매개변수 이름과 실제 매개변수 이름이 서로 다른 기호일 뿐입니다. 프로그램을 실행하는 동안 메모리의 저장 공간이 가장 중요합니다. 변수 이름이 다르다고 해서 차지하는 메모리 저장 공간이 다르다는 의미는 아닙니다.
일반적으로 두 호출의 근본적인 차이점은 값이 전달되는지 주소가 전달되는지가 아니라(결국 주소도 값임) 형식 매개변수와 실제 매개변수가 동일한 메모리를 점유하는지 여부입니다. 공간. 실제로 C/C++에서 전달하는 포인터 매개변수도 값 호출입니다. 믿을 수 없다면 다음 C 코드를 시도해 보세요!
#include<stdio.h> void swap(int *a1,int *b1){ int *t=a1; a1=b1; b1=t; } int main(){ int x1=100; int x2=200; int *a=&x1; int *b=&x2; printf("%d %d\n",*a,*b); swap(a,b); printf("%d %d\n",*a,*b); return 0; }
그러나 C/C++는 참조에 의해 호출됩니다. 이것은 C/C++에서 reference라고 불리는 변수 선언 방법입니다. 여기서 ra는 a입니다. 메모리에서는 둘 사이에 차이가 없으며 동일한 메모리 공간을 차지합니다. 참조(별칭)에 의한 매개변수 전달은 참조 호출의 특성을 따릅니다. void swap(int &a1,int &b1);의 연산 결과를 시험해 볼 수 있습니다.
이 기사를 통해 Java 메소드 매개변수가 참조로 호출되는지 아니면 값으로 호출되는지 알아야 합니다.
Java 메소드 매개변수가 참조 호출인지 값 호출인지에 대한 더 많은 기사를 보려면 PHP 중국어 웹사이트를 주목하세요!