Java의 String 객체는 변경 가능합니까?
한 친구가 StackOverflow에 질문을 올려 이렇게 물었습니다.
우리 모두 알고 있듯이 Java의 String 객체는 불변이지만 다음 코드를 살펴보겠습니다.
String s1 = "Hello World"; String s2 = "Hello World"; String s3 = s1.substring(6); System.out.println(s1); // Hello World System.out.println(s2); // Hello World System.out.println(s3); // World Field field = String.class.getDeclaredField("value"); field.setAccessible(true); char[] value = (char[])field.get(s1); value[6] = 'J'; value[7] = 'a'; value[8] = 'v'; value[9] = 'a'; value[10] = '!'; System.out.println(s1); // Hello Java! System.out.println(s2); // Hello Java! System.out.println(s3); // World
이 코드는 왜 이렇게 실행되나요? s1과 s2의 값은 변경되는데 s3의 값은 변경되지 않는 이유는 무엇입니까?
답변 #1:
String 객체는 불변이지만 이는 공개 메서드를 호출하여 그 값을 변경할 수 없다는 의미입니다.
위 코드는 리플렉션 메커니즘을 통해 일반 API를 우회합니다. 이러한 방식으로 열거형 값을 변경할 수도 있고 Integer 유형이 자동 박싱될 때 사용되는 조회 테이블을 변경할 수도 있습니다.
여기서 s1과 s2는 동일한 내부 문자열 객체를 가리키므로 값이 모두 변경됩니다. 다른 답변에서 언급했듯이 이는 컴파일러에 의해 구현됩니다.
s3가 변경되지 않은 이유는 정말 놀랍습니다. 저는 s3과 s1이 동일한 값 배열을 공유한다고 생각했습니다(Java 7u6 이전 버전에서는 실제로 그랬습니다). 그러나 String 클래스의 소스 코드를 보면 하위 문자열 객체의 값 배열이 원래 문자열 객체에서 복사되는 것을 볼 수 있습니다(Arrays.copyOfRange(..) 메서드 사용). 이것이 s3가 변경되지 않은 이유입니다.
악성 코드가 이러한 유형의 작업을 수행하지 못하도록 SecurityManager를 설치할 수 있습니다. 그러나 일부 라이브러리의 구현은 이러한 반사 기술(예: ORM 도구, AOP 라이브러리 등)에 의존한다는 점에 유의해야 합니다.
답장 시작 부분에 String 객체는 실제로 불변이 아니라 단지 "불변으로 보인다"고 썼습니다. 이는 독자가 String 클래스의 현재 버전이 액세스 제한 측면에서 부주의하다고 생각하도록 오해할 수 있지만 실제로 값 배열은 private 및 final 수정자를 사용합니다. 따라서 개발자는 주의를 기울여야 합니다. Java에서는 배열을 불변으로 선언할 수 없으며 올바른 액세스 한정자가 사용되더라도 클래스 외부에 노출될 수 없습니다.
이 주제는 매우 중요하므로 몇 가지 고급 내용을 읽어 보시기 바랍니다. 2009 JavaZone 컨퍼런스에서 반사 기술에 대한 Heinz Kabutz의 미친 연설 이 기사에서는 반사 작업의 일반적인 문제와 기타 반사 관련 문제를 다룹니다. 기술 콘텐츠. 이 글은 매우 훌륭하고 매우 미쳤습니다.
이 문서에서는 반사 기법이 특정 시나리오에서 유용한 이유를 설명하지만 대부분의 경우 사용을 피해야 합니다.
답변 #2:
Java에서는 문자열 유형의 두 변수가 동일한 문자열로 초기화되면 두 변수에 동일한 개체 참조가 할당됩니다. 이것이 "Test1==Test2" 표현식이 true를 반환하는 이유입니다.
String Test1="Hello World"; String Test2="Hello World"; System.out.println(test1==test2); // true
Test3은 substring() 메서드로 생성된 새로운 String 개체입니다. Test1과 동일한 값 배열을 공유하지 않습니다. (참고: 원저자의 사무적인 오류로 인해 아래 그림의 변수 test1과 test3의 첫 글자는 대문자로 표기되지 않았습니다. 독자 여러분의 주의를 바랍니다.)
Reflection 기술을 통해 String 객체에 접근할 수 있으며, 값 배열의 포인터를 얻습니다.
Field field = String.class.getDeclaredField("value"); field.setAccessible(true);
이 값 배열의 값을 변경하면 배열 포인터를 보유하는 모든 String 개체의 값이 변경될 수 있으므로 값은 Test1과 Test2의 변경되었습니다. 그러나 Test3은 substring() 메소드에 의해 생성된 새로운 String 객체이므로 해당 값은 변경되지 않았습니다.

핫 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
