Java 개발에는 겉보기에는 간단해 보이는데 인터넷에 떠도는 주제와 질문이 참 많습니다. 즉, equals()와 == 연산자의 차이점이 무엇인가요?
오늘의 콘텐츠 소개는 약 9분 정도 걸립니다
Pictures
이 차이점을 더 잘 이해하기 위해 예를 살펴보겠습니다.
String str1 = new String("Hello");String str2 = new String("Hello");System.out.println(str1.equals(str2)); // 输出 trueSystem.out.println(str1 == str2); // 输出 false
이 예에서는 두 문자열의 내용은 동일하지만 메모리의 주소는 다릅니다. 따라서 .equals() 메서드를 사용하여 내용을 비교하면 true를 반환하고, "==" 연산자를 사용하여 주소를 비교하면 false를 반환합니다
Java의 기본, Java의 모든 클래스는 기본적으로 Object 클래스를 상속한다는 것을 알아야 합니다. Object 클래스에는 .equals() 메서드가 있습니다
public boolean equals(Object obj) {return (this == obj);}
코드에서 .equals() 메서드를 찾을 수 있습니다. 기본적으로 비교를 위해 == 연산자를 사용합니다. 하위 클래스가 equals() 메서드를 재정의하지 않은 경우 == 연산자와 equals() 메서드를 사용하면 결과는 정확히 동일합니다. 두 메모리 주소가 서로 다른지 비교하는 데 사용됩니다. 객체는 동일합니다.
그러나 실제 상황은 equals() 메서드를 재정의하는 클래스가 많다는 것입니다. 이는 메모리 주소 비교 요구 사항이 엄격하고 모든 실제 시나리오의 요구 사항을 충족하지 않기 때문입니다. 예를 들어 String 클래스는 비교할 때 대부분은 내용이 같은지 판단하고 싶은 것이지, 메모리 주소가 같은지(객체인지)는 별로 알고 싶지 않습니다.
Java 문자열 상수 풀에 대한 심층 연구 기사에서 우리는 Java 가상 머신이 메모리 활용을 최적화하고 성능을 향상시키기 위해 문자열을 위한 별도의 공간, 즉 문자열 상수 풀을 할당한다는 것을 배웠습니다.
new 키워드를 사용하는 대신 String s = "Hello"를 사용하여 문자열 객체를 생성하는 것이 좋습니다. new를 사용하려면 힙에 추가 메모리 공간 할당이 필요하기 때문입니다.
Jdk11의 String 클래스의 equals() 메서드
public boolean equals(Object anObject) { //如果是同一个对象(即两个引用指向内存中的同一块地址),则直接返回trueif (this == anObject) {return true;} //如果是String类型的实例if (anObject instanceof String) { //Object类型的对象强制转换为String类型String aString = (String)anObject;//如果当前字符串对象和传入的字符串对象的编码方式相同if (coder() == aString.coder()) { //如果当前字符串和传入的字符串都是Latin1编码,则调用StringLatin1类的equals方法进行比较;如果其中一个或两个字符串是UTF16编码,则调用StringUTF16类的equals方法进行比较return isLatin1() ? StringLatin1.equals(value, aString.value): StringUTF16.equals(value, aString.value);}}return false;}
특별 참고 사항: Latin1(ISO 8859-1이라고도 함) 및 UTF-16(유니코드) 변환 형식 16비트)는 두 가지 다른 문자 인코딩 방법입니다
Latin1과 UTF-16은 두 가지 인코딩 방법이지만 차이는 크지 않습니다. Java8 Java11의 equals() 메소드의 소스 코드는 JDK8
@HotSpotIntrinsicCandidatepublic static boolean equals(byte[] value, byte[] other) {if (value.length == other.length) {int len = value.length >> 1;for (int i = 0; i
1.2의 equals() 메소드와 다릅니다. 예시 설명
public boolean equals(Object anObject) {// 如果是同一个对象(即两个引用指向内存中的同一块地址),则直接返回trueif (this == anObject) {return true;}// 如果是String类型的实例if (anObject instanceof String) {////Object类型的对象强制转换为String类型String anotherString = (String)anObject;int n = value.length;// 如果字符串长度相等if (n == anotherString.value.length) {char v1[] = value;char v2[] = anotherString.value;int i = 0;// 判断每个字符是否相等while (n-- != 0) {if (v1[i] != v2[i])return false;i++;}return true;}}return false;}
출력 결과는 무엇입니까? String 클래스의 equals 메소드는 문자열 객체의 내용이 모두 "Hello"이므로 결과가 true인지 여부를 결정합니다. 예 2:
new String("hello").equals("hello")
출력 결과는 무엇입니까?
== 연산자가 비교한 개체 주소가 동일한지 여부, = =왼쪽이 힙에 생성된 개체이고, 오른쪽이 문자열 상수 풀 개체이므로 내용은 동일하지만 주소가 동일하지 않습니다. 결과는 false입니다
예제 3:
new String("hello") == "hello";
출력 결과는 무엇입니까?
새 개체 완전히 다른 메모리 주소여야 하므로 결과는 false입니다
예제 4:
new String("hello") == new String("hello");
무엇입니까? 출력 결과는?
h와 ello는 모두 문자열 상수 풀에 있으므로 컴파일러가 + 연산자를 만나면 자동으로 이를 hello로 최적화하여 결과가 true가 됩니다.
예 5:
"hello" == "h"+"ello"
출력은 무엇입니까? result?
new String("hello") 실행되면 먼저 문자열 상수 pool에 객체가 생성된 후 intern() 메서드를 실행하면 힙에 객체가 생성되는 것을 발견합니다. hello'는 문자열 상수 풀에 이미 존재하므로 문자열 상수 풀에 있는 객체 참조를 직접 반환한 다음 상수 풀에 있는 문자열 'hello' 비교와 비교하므로 결과는 true입니다
심층 분석 String.intern()은 이미 그 이유를 소개했습니다
2. String 클래스의 .contentEquals() 메서드: 이 메서드는 문자열이 임의의 문자 시퀀스(예: StringBuffer, StringBuilder, String, CharSequence)와 동일한지 비교하는 데 사용할 수 있습니다.new String("hello").intern() == "hello"
Objects.equals("Hello", new String("Hello")); // 返回 true
위 내용은 Java에서 equals()와 ==의 차이점과 사용법의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!