Java에서 값으로 전달과 참조로 전달의 차이점은 무엇입니까
차이점: 1. 값 전송은 복사본을 생성하지만 참조 전송은 복사본을 생성하지 않습니다. 2. 값 전송에서는 함수에서 원본 객체를 변경할 수 없지만 참조 전송에서는 원본 객체를 변경할 수 있습니다. 기능. 값으로 전달한다는 것은 함수를 호출할 때 실제 매개변수의 복사본이 함수에 전달된다는 의미입니다. 따라서 함수에서 매개변수가 수정되면 실제 매개변수는 영향을 받지 않습니다. 참조로 전달한다는 것은 실제 매개변수가 복사된다는 의미입니다. 함수를 호출할 때 주소는 함수에 직접 전달되므로 함수의 매개변수를 수정하면 실제 매개변수에 영향을 미칩니다.
이 튜토리얼의 운영 환경: windows7 시스템, java8 버전, DELL G3 컴퓨터.
실제 및 형식 매개변수
우리 모두는 Java에서 메소드를 정의할 때 매개변수를 정의할 수 있다는 것을 알고 있습니다. 예를 들어, Java의 주요 메소드인 public static void main(String[ ] args), 여기서 args는 매개변수입니다. 프로그래밍 언어에서는 매개변수를 형식 매개변수와 실제 매개변수로 구분합니다.
정식 매개변수: 함수 이름과 함수 본문을 정의할 때 사용되는 매개변수입니다. 함수 호출 시 전달되는 매개변수를 받는 것이 목적입니다.
실제 매개변수: 매개변수화된 함수를 호출할 때 호출하는 함수와 호출되는 함수 사이에 데이터 전송 관계가 있습니다. 호출 함수에서 함수를 호출할 때 함수 이름 뒤에 괄호 안의 매개변수를 "실제 매개변수"라고 합니다
간단한 예:
public static void main( String[ ] args) { ParamTest pt = new ParamTest(); pt.sout( "Hollis");//实际参数为Hollis } public void sout( String name) {/!形式参数为name system.out.println(name); }
실제 매개변수는 매개변수화된 메소드를 호출할 때 실제로 전달되는 매개변수이고, Formal 매개변수는 매개변수는 실제 매개변수의 내용을 받는 데 사용되는 매개변수입니다.
값 전달 및 참조 전달
위에서 언급했듯이 매개변수화된 함수를 호출하면 실제 매개변수가 형식 매개변수로 전달됩니다. 그러나 프로그래밍 언어에서는 이 전송 프로세스에서 전송의 두 가지 경우, 즉 값에 의한 전송과 참조에 의한 전송이 있습니다. 값에 의한 전달과 참조에 의한 전달이 프로그래밍 언어에서 어떻게 정의되고 구별되는지 살펴보겠습니다.
값 전달은 함수 호출 시 실제 매개변수를 복사하여 함수에 전달한다는 의미로, 함수에서 매개변수가 수정되더라도 실제 매개변수에는 영향을 주지 않습니다.
참조로 전달은 함수를 호출할 때 실제 매개변수의 주소를 함수에 직접 전달하는 것을 의미합니다. 그런 다음 함수의 매개변수를 수정하면 실제 매개변수에 영향을 미칩니다.
위 개념을 사용하여 코드를 작성하고 연습하여 Java에서 값 전달인지 참조 전달인지 확인할 수 있습니다. 따라서 가장 간단한 코드 조각이 나왔습니다.
public static void main( String[] args) { ParamTest pt = new ParamTest(); int i = 10; pt.pass(i); System.out.println( "print in main , i is " +i); } public void pass(int j){ j = 20; system.out.println( "print in pass , j is " + j); }
위 코드에서 우리는 매개변수 j는 pass 메소드에서 수정된 후 매개변수의 값이 pass 메소드와 main 메소드에서 각각 인쇄됩니다. 출력 결과는 다음과 같습니다.
print in pass , j is 20 print in main , i is 10
pass 메소드 내에서 i 값을 수정해도 실제 매개변수 i 값은 변경되지 않는 것을 알 수 있습니다. 따라서 위의 정의에 따르면 누군가는 다음과 같은 결론에 도달했습니다. Java의 메소드 전달은 값 전달입니다.
그런데 곧 몇몇 분들이 질문을 주셨는데요(웃음, 성급히 결론 내리지는 마세요.). 그런 다음 다음 코드를 이동합니다.
public static void main(String[ ] args) { ParamTest pt = new ParamTest(); User hollis = new User(); hollis.setName( "Hollis"); hollis.setGender("Male"); pt.pass(hollis); system.out.println( "print in main , user is " + hollis);}public void pass(User user) { user.setName( "hollischuang"); System.out.println( "print in pass , user is " + user);}
도 전달 방법이며 매개변수 값도 전달 방법 내에서 수정됩니다. 출력 결과는 다음과 같습니다.
print in pass , user is User{name='hollischuang', gender='Male '} print in main , user is User{name='hollischuang' , gender='Male '}
pass 메소드가 실행된 후 실제 매개변수의 값이 변경됩니다. 위의 참조로 전달의 정의에 따라 실제 매개변수의 값이 변경됩니다. 참조로 전달? 따라서 위의 두 가지 코드를 기반으로 누군가는 새로운 결론에 도달했습니다. Java 메서드에서 일반 유형을 전달할 때는 값으로 전달되고 객체 유형을 전달할 때는 참조로 전달됩니다.
그러나 이 진술은 여전히 잘못된 것입니다. 믿을 수 없다면 매개변수 유형이 객체인 다음 매개변수 전송을 살펴보세요.
public static void main( string[] args) { ParamTest pt = new ParamTest(); string name = "Hollis"; pt.pass(name ) ; System.out.println( "print in main , name is " + name); } public void pass(string name) { name = "hollischuang"; system.out.println( "print in pass , name is " + name); }
위 코드의 출력 결과는
print in pass , name is hollischuangprint in main , name is Hollis
객체도 전달되는데 설명은 무엇입니까? 원래 매개 변수의 값이 수정되지 않았습니다. 객체가 다시 값으로 전달됩니까?
Java에서 값 전달
위에서 세 가지 예를 제시했지만 이로 인해 많은 초보자가 발생했습니다. 심지어 많은 고급 프로그래머들도 Java 유형 전달에 대해 혼란스러운 이유를 갖고 있습니다. 사실 제가 말씀드리고 싶은 것은 위의 개념은 틀린 것이 아니지만 코드 예시에는 문제가 있다는 것입니다. 자, 제가 여러분을 위해 개념의 핵심 사항을 개괄적으로 설명하고 몇 가지 적절한 예를 들어보겠습니다.
값 전달은 함수 호출 시 실제 매개변수를 복사하여 함수에 전달한다는 의미로, 함수에서 매개변수가 수정되더라도 실제 매개변수에는 영향을 주지 않습니다.
참조로 전달은 함수를 호출할 때 실제 매개변수의 주소를 함수에 직접 전달하는 것을 의미합니다. 그런 다음 함수의 매개변수를 수정하면 실제 매개변수에 영향을 미칩니다.
그래서, 값 전달과 참조 전달의 차이점에 대한 핵심 사항을 요약해 보겠습니다.
값으로 전달 | 참조로 전달 | |
---|---|---|
근본적인 차이점 | 복사본을 생성합니다 | 복사본을 생성하지 않습니다 |
모두 | 기능은 변경할 수 없습니다. 원본 개체 | 기능은 원본 개체 |
我们上面看过的几个pass的例子中,都只关注了实际参数内容是否有改变。如传递的是User对象,我们试着改变他的name属性的值,然后检查是否有改变。其实,在实验方法上就错了,当然得到的结论也就有问题了。
为什么说实验方法错了呢?这里我们来举一个形象的例子。再来深入理解一下值传递和引用传递,然后你就知道为啥错了。
你有一把钥匙,当你的朋友想要去你家的时候,如果你直接把你的钥匙给他了,这就是引用传递。这种情况下,如果他对这把钥匙做了什么事情,比如他在钥匙上刻下了自己名字,那么这把钥匙还给你的时候,你自己的钥匙上也会多出他刻的名字。
你有一把钥匙,当你的朋友想要去你家的时候,你复刻了一把新钥匙给他,自己的还在自己手里,这就是值传递。这种情况下,他对这把钥匙做什么都不会影响你手里的这把钥匙。
但是,不管上面那种情况,你的朋友拿着你给他的钥匙,进到你的家里,把你家的电视砸了。那你说你会不会受到影响?而我们在pass方法中,改变user对象的name属性的值的时候,不就是在“砸电视”么。
还拿上面的一个例子来举例,我们真正的改变参数,看看会发生什么?
public static void main(String[ ] args){ ParamTest pt = new ParamTest(); User hollis = new User(); hollis.setName( "Hollis"); hollis.setGender("Male" ); pt.pass(hollis); system.out.println("print in main , user is " + hollis); public void pass(User user) { user = new User(); user.setName( "hollischuang"); user.setGender( "Male"); system.out.println( "print in pass , user is " + user);
上面的代码中,我们在pass方法中,改变了user对象,输出结果如下:
print in pass , user is User{name='hollischuang ' , gender='Male '} print in main , user is User{name='Hollis', gender= 'Male '}
我们来画一张图,看一下整个过程中发生了什么,然后我再告诉你,为啥Java中只有值传递。
稍微解释下这张图,当我们在main中创建一个User对象的时候,在堆中开辟一块内存,其中保存了name和gender等数据。然后hollis持有该内存的地址ex123456(图1)。当尝试调用pass方法,并且hollis作为实际参数传递给形式参数user的时候,会把这个地址ex123456交给user,这时,user也指向了这个地址(图2)。然后在pass方法内对参数进行修改的时候,即user = newUser();,会重新开辟一块 eX456789的内存,赋值给user。后面对user的任何修改都不会改变内存eX123456的内容(图3)。
上面这种传递是什么传递?肯定不是引用传递,如果是引用传递的话,在user=new User()的时候,实际参数的引用也应该改为指向eX456789,但是实际上并没有。
通过概念我们也能知道,这里是把实际参数的引用的地址复制了一份,传递给了形式参数。所以,上面的参数其实是值传递,把实参对象引用的地址当做值传递给了形式参数。
我们再来回顾下之前的那个“砸电视”的例子,看那个例子中的传递过程发生了什么。
同样的,在参数传递的过程中,实际参数的地址eX1213456被拷贝给了形参,只是,在这个方法中,并没有对形参本身进行修改,而是修改的形参持有的地址中存储的内容。
所以,值传递和引用传递的区别并不是传递的内容。而是实参到底有没有被复制一份给形参。在判断实参内容有没有受影响的时候,要看传的的是什么,如果你传递的是个地址,那么就看这个地址的变化会不会有影响,而不是看地址指向的对象的变化。就像钥匙和房子的关系。
那么,既然这样,为啥上面同样是传递对象,传递的String对象和User对象的表现结果不一样呢?我们在pass方法中使用name = “hollischuang”;试着去更改name的值,阴差阳错的直接改变了name的引用的地址。因为这段代码,会new一个String,在把引用交给name,即等价于name =new String(“hollischuang”);。而原来的那个”Hollis”字符串还是由实参持有着的,所以,并没有修改到实际参数的值。
所以说,Java中其实还是值传递的,只不过对于对象参数,值的内容是对象的引用。
总结
无论是值传递还是引用传递,其实都是一种求值策略(Evaluation strategy)。在求值策略中,还有一种叫做按共享传递。其实Java中的参数传递严格意义上说应该是按共享传递。
공유하여 전달한다는 것은 함수가 호출될 때 실제 매개변수의 주소 복사본이 함수에 전달된다는 의미입니다(실제 매개변수가 스택에 있는 경우 값이 직접 복사됩니다). 함수 내에서 매개변수를 연산할 경우, 연산하기 전에 특정 값을 찾기 위해 주소를 복사해야 합니다. 값이 스택에 있는 경우 값의 직접 복사본이므로 함수 내의 매개변수에 대한 작업은 외부 변수에 영향을 주지 않습니다. 원본이 힙에 있는 원래 값의 주소인 경우 작업을 수행하기 전에 해당 주소를 기반으로 힙에서 해당 위치를 찾아야 합니다. 주소 복사본이 전달되므로 함수 내의 값에 대한 작업이 외부 변수에 표시됩니다.
간단히 말하면 Java에서의 전송은 값에 의한 것이며 이 값은 실제로 객체에 대한 참조입니다.
공유에 의한 전달은 사실 가치에 의한 전달의 특별한 경우일 뿐입니다. 따라서 Java를 전달하는 것은 공유를 통한 전달이거나 Java를 전달하는 것은 값을 통한 전달이라고 말할 수 있습니다.
그래서 함수 내부의 매개변수를 조작해도 외부 변수에는 영향을 미치지 않습니다. 원본이 힙에 있는 원래 값의 주소인 경우 작업을 수행하기 전에 해당 주소를 기반으로 힙에서 해당 위치를 찾아야 합니다. 주소 복사본이 전달되므로 함수 내의 값에 대한 작업이 외부 변수에 표시됩니다.
간단히 말하면 Java에서의 전송은 값에 의한 것이며 이 값은 실제로 객체에 대한 참조입니다.
그리고 공유를 전달하는 것은 실제로 가치를 전달하는 특별한 경우일 뿐입니다. 따라서 Java를 전달하는 것은 공유를 통한 전달이거나 Java를 전달하는 것은 값을 통한 전달이라고 말할 수 있습니다.
더 많은 프로그래밍 관련 지식을 보려면 프로그래밍 교육을 방문하세요! !
위 내용은 Java에서 값으로 전달과 참조로 전달의 차이점은 무엇입니까의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

핫 AI 도구

Undresser.AI Undress
사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover
사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool
무료로 이미지를 벗다

Clothoff.io
AI 옷 제거제

AI Hentai Generator
AI Hentai를 무료로 생성하십시오.

인기 기사

뜨거운 도구

메모장++7.3.1
사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전
중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기
강력한 PHP 통합 개발 환경

드림위버 CS6
시각적 웹 개발 도구

SublimeText3 Mac 버전
신 수준의 코드 편집 소프트웨어(SublimeText3)

뜨거운 주제











Java의 난수 생성기 안내. 여기서는 예제를 통해 Java의 함수와 예제를 통해 두 가지 다른 생성기에 대해 설명합니다.

Java의 Weka 가이드. 여기에서는 소개, weka java 사용 방법, 플랫폼 유형 및 장점을 예제와 함께 설명합니다.

Java의 Smith Number 가이드. 여기서는 정의, Java에서 스미스 번호를 확인하는 방법에 대해 논의합니다. 코드 구현의 예.

이 기사에서는 가장 많이 묻는 Java Spring 면접 질문과 자세한 답변을 보관했습니다. 그래야 면접에 합격할 수 있습니다.

Java 8은 스트림 API를 소개하여 데이터 컬렉션을 처리하는 강력하고 표현적인 방법을 제공합니다. 그러나 스트림을 사용할 때 일반적인 질문은 다음과 같은 것입니다. 기존 루프는 조기 중단 또는 반환을 허용하지만 스트림의 Foreach 메소드는이 방법을 직접 지원하지 않습니다. 이 기사는 이유를 설명하고 스트림 처리 시스템에서 조기 종료를 구현하기위한 대체 방법을 탐색합니다. 추가 읽기 : Java Stream API 개선 스트림 foreach를 이해하십시오 Foreach 메소드는 스트림의 각 요소에서 하나의 작업을 수행하는 터미널 작동입니다. 디자인 의도입니다

Java의 TimeStamp to Date 안내. 여기서는 소개와 예제와 함께 Java에서 타임스탬프를 날짜로 변환하는 방법에 대해서도 설명합니다.

캡슐은 3 차원 기하학적 그림이며, 양쪽 끝에 실린더와 반구로 구성됩니다. 캡슐의 부피는 실린더의 부피와 양쪽 끝에 반구의 부피를 첨가하여 계산할 수 있습니다. 이 튜토리얼은 다른 방법을 사용하여 Java에서 주어진 캡슐의 부피를 계산하는 방법에 대해 논의합니다. 캡슐 볼륨 공식 캡슐 볼륨에 대한 공식은 다음과 같습니다. 캡슐 부피 = 원통형 볼륨 2 반구 볼륨 안에, R : 반구의 반경. H : 실린더의 높이 (반구 제외). 예 1 입력하다 반경 = 5 단위 높이 = 10 단위 산출 볼륨 = 1570.8 입방 단위 설명하다 공식을 사용하여 볼륨 계산 : 부피 = π × r2 × h (4
