Java java지도 시간 Java에서 객체가 동일한지 확인하기 위해 equals() 메소드를 사용하는 방법에 대한 튜토리얼

Java에서 객체가 동일한지 확인하기 위해 equals() 메소드를 사용하는 방법에 대한 튜토리얼

Feb 03, 2017 pm 01:08 PM
java equals

Object 클래스의 equals 메소드는 한 객체가 다른 객체와 동일한지 여부를 감지하는 데 사용됩니다. Object 클래스에서 이 메서드는 두 객체가 동일한 참조를 가지고 있는지 여부를 확인합니다. 두 객체가 동일한 참조를 가지고 있으면 두 객체는 ​​동일해야 합니다. 이러한 관점에서 이를 기본 작업으로 만드는 것이 합리적입니다. 그러나 대부분의 클래스에서는 이러한 판단이 의미가 없습니다. 예를 들어 두 개의 PrintStream을 이런 방식으로 비교하는 것은 완전히 의미가 없습니다. 그러나 두 객체의 상태가 동일한지 확인해야 하는 경우가 종종 있습니다. 두 객체의 상태가 동일하면 두 객체가 동일한 것으로 간주됩니다. 따라서 사용자 정의 클래스에서는 같음 비교를 재정의해야 합니다.

다음은 완벽한 equals() 메소드 작성을 위한 제안 사항입니다.

(1) 명시적 매개변수의 이름은 otherObject이며 나중에 other라는 변수로 변환해야 합니다.

(2) this와 otherObject가 동일한 객체를 참조하는지 확인하세요.

if(this==otherObject) return true;
로그인 후 복사

이 문은 단지 최적화일 뿐입니다. 실제로 이는 흔히 취하는 형태이다. 왜냐하면 이 방정식을 계산하는 것이 클래스의 필드를 하나씩 비교하는 것보다 훨씬 저렴하기 때문입니다.

(3) otherObject가 null인지 확인합니다. null인 경우 false를 반환합니다. 이 테스트는 매우 필요합니다.

if(otherObject==null) return false;
로그인 후 복사

(4) this와 otherObject를 비교하여 동일한 클래스에 속하는지 확인합니다. 각 하위 클래스에서 equals의 의미가 변경되면 getClass()를 사용하여 이를 감지합니다. 자신을 Target 클래스

if(getClass()!=otherObject.getClass()) return false;
로그인 후 복사

로 취급합니다. 모든 하위 클래스가 동일한 의미 체계를 갖는 경우, instanceof를 사용하여

if(!(otherObject instanceof ClassName)) return false;
로그인 후 복사

를 감지합니다. (5) otherObject를 다음으로 변환합니다. 해당 유형의 변수 :

ClassName other=(ClassName)otherObject;
로그인 후 복사

(6) 이제 비교해야 할 모든 필드를 비교 시작합니다. 기본 유형 필드를 비교하려면 ==를 사용하고 객체 필드를 비교하려면 같음을 사용하세요. 모든 필드가 일치하면 true를 반환하고, 그렇지 않으면 false를 반환합니다.

return field1==other.field1&&field2.equals(other.field2)
로그인 후 복사

equals가 하위 클래스에서 재정의된 경우 super.equals(other) 호출을 포함해야 합니다. 테스트가 실패하면 동등성은 불가능합니다. 슈퍼클래스의 필드가 동일하면 서브클래스의 인스턴스 필드가 비교됩니다.

배열 유형 필드의 경우 정적 Arrays.equals 메서드를 사용하여 해당 요소가 동일한지 확인할 수 있습니다.

몇 가지 문자열 비교 예를 살펴보겠습니다.

String a = "abc";
String b = "abc";
String c = new String("abc");
String d = new String("abc");
System.out.println(a == b); // true 因为JAVA中字符串常量是共享的,只有一个拷贝
System.out.println(a == c); // false a和c属于2个不同的对象
System.out.println(a.equals(c)); // true 由于String对象的equals方法比较的是对象中的值,所以返回true。(和Object的equals方法不同)
System.out.println(c==d); // false c和d虽然对象内的值相同,但属于2个不同的对象,所以不相等
System.out.println(c.equals(d)); // true
로그인 후 복사

간단히 말하면 문자열 상수를 비교할 때 같음은 비교하려는 경우와 동일한 결과를 반환합니다. 문자열 객체의 값을 사용할 때와 같습니다.

equal 사용 예 보기:

package chapter05.EqualsTest;
  
import java.util.*;
  
public class EqualsTest {
 public static void main(String[] args) {
  Employee alice1 = new Employee("Alice Adams", 75000, 1987, 12, 15);
  Employee alice2 = alice1; // reference the same object
  Employee alice3 = new Employee("Alice Adams", 75000, 1987, 12, 15);
  Employee bob = new Employee("Bob Brandson", 50000, 1989, 10, 1);
  
  System.out.println("alice1 == alice2: " + (alice1 == alice2));
  
  System.out.println("alice1 == alice3: " + (alice1 == alice3));
  
  System.out.println("alice1.equals(alice3): " + (alice1.equals(alice3)));
  
  System.out.println("alice1.equals(bob): " + (alice1.equals(bob)));
  
  System.out.println(bob.toString());
 }
}
  
class Employee {
 public Employee(String n, double s, int year, int month, int day) {
  name = n;
  salary = s;
  GregorianCalendar calendar = new GregorianCalendar(year, month, day);
  hireDay = calendar.getTime();
 }
  
 public String getName() {
  return name;
 }
  
 public double getSalary() {
  return salary;
 }
  
 public Date getHireDay() {
  return hireDay;
 }
  
 public void raiseSalary(double byPercent) {
  double raise = salary * byPercent / 100;
  salary += raise;
 }
  
 @Override
 public boolean equals(Object otherObject) {
  // a quick test to see if the objects are identical
  if (this == otherObject)
   return true;
  
  // must return false if the explicit parameter is null
  if (otherObject == null)
   return false;
  
  // if the classed don't match,they can't be equal
  if (getClass() != otherObject.getClass())
   return false;
  
  // now we know otherObject is a non-null Employee
  Employee other = (Employee) otherObject;
  
  // test whether the fields hava identical values
  return name.equals(other.name) && salary == other.salary
    && hireDay.equals(other.hireDay);
  
 }
  
 @Override
 public int hashCode() {
  return 7 * name.hashCode() + 11 * new Double(salary).hashCode() + 13
    * hireDay.hashCode();
 }
  
 @Override
 public String toString() {
  return getClass().getName() + "[name=" + name + ",salary=" + salary
    + ",hireDay=" + hireDay + "]";
 }
  
 private String name;
 private double salary;
 private Date hireDay;
}
  
class Manager extends Employee {
 public Manager(String n, double s, int year, int month, int day) {
  super(n, s, year, month, day);
  bouns = 0;
 }
  
 @Override
 public double getSalary() {
  double baseSalary = super.getSalary();
  return baseSalary + bouns;
 }
  
 public void setBouns(double b) {
  bouns = b;
 }
  
 @Override
 public boolean equals(Object otherObject) {
  if (!super.equals(otherObject))
   return false;
  Manager other = (Manager) otherObject;
  // super equals checked that this and other belong to the same class
  return bouns == other.bouns;
 }
  
 @Override
 public int hashCode() {
  return super.hashCode() + 17 * new Double(bouns).hashCode();
 }
  
 @Override
 public String toString() {
  return super.toString() + "[bouns=" + bouns + "]";
 }
  
 private double bouns;
}
로그인 후 복사

더 자세히
다음은 "클래스가 equals()를 포함하는지 여부에 따라 2가지 범주로 나뉩니다. 방법".
(1) 클래스가 equals() 메소드를 다루지 않는 경우, equals()를 통해 두 객체를 비교할 때 실제로 두 객체가 동일한 객체인지 비교합니다. 이때 "=="를 통해 두 객체를 비교하는 것과 같습니다.
(2) 클래스의 equals() 메소드를 재정의하여 equals()가 다른 메소드를 통해 두 객체가 동일한지 비교할 수 있습니다. 일반적인 접근 방식은 두 객체의 내용이 동일하면 equals() 메서드가 true를 반환하고, 그렇지 않으면 fasle을 반환하는 것입니다.
다음은 위의 두 가지 상황을 예시로 나타낸 것입니다.
1. "equals() 메서드가 적용되지 않습니다" 상황
코드는 다음과 같습니다(EqualsTest1.java):

import java.util.*;
import java.lang.Comparable;
 
/**
 * @desc equals()的测试程序。
 
 */
public class EqualsTest1{
 
 public static void main(String[] args) {
  // 新建2个相同内容的Person对象,
  // 再用equals比较它们是否相等
  Person p1 = new Person("eee", 100);
  Person p2 = new Person("eee", 100);
  System.out.printf("%s\n", p1.equals(p2));
 }
 
 /**
  * @desc Person类。
  */
 private static class Person {
  int age;
  String name;
 
  public Person(String name, int age) {
   this.name = name;
   this.age = age;
  }
 
  public String toString() {
   return name + " - " +age;
  }
 }
}
로그인 후 복사

실행 결과:

false
로그인 후 복사


결과 분석
p1.equals(p2)를 사용하여 "p1과 p2가 같은지 비교"합니다. 실제로는 Object.java의 equals() 메소드, 즉 (p1==p2)가 호출된다. "p1과 p2가 동일한 객체인지"를 비교합니다.
p1과 p2의 정의를 통해 우리는 동일한 내용을 가지고 있지만 서로 다른 두 개체라는 것을 알고 있습니다! 따라서 반환 결과는 false입니다.

2. "equals() 메서드를 재정의한 경우"
위의 EqualsTest1.java를 수정합니다: equals() 메서드를 재정의합니다.
코드는 다음과 같습니다(EqualsTest2.java):

import java.util.*;
import java.lang.Comparable;
 
/**
 * @desc equals()的测试程序。
 */
public class EqualsTest2{
 
 public static void main(String[] args) {
  // 新建2个相同内容的Person对象,
  // 再用equals比较它们是否相等
  Person p1 = new Person("eee", 100);
  Person p2 = new Person("eee", 100);
  System.out.printf("%s\n", p1.equals(p2));
 }
 
 /**
  * @desc Person类。
  */
 private static class Person {
  int age;
  String name;
 
  public Person(String name, int age) {
   this.name = name;
   this.age = age;
  }
 
  public String toString() {
   return name + " - " +age;
  }
 
  /**
   * @desc 覆盖equals方法
   */
  @Override
  public boolean equals(Object obj){
   if(obj == null){
    return false;
   }
 
   //如果是同一个对象返回true,反之返回false
   if(this == obj){
    return true;
   }
 
   //判断是否类型相同
   if(this.getClass() != obj.getClass()){
    return false;
   }
 
   Person person = (Person)obj;
   return name.equals(person.name) && age==person.age;
  }
 }
}
로그인 후 복사

실행 결과:

true
로그인 후 복사

결과 분석:
EqualsTest2.java에서 Person을 다시 작성했습니다. equals() 함수: 두 Person 객체의 이름과 나이가 같으면 true를 반환합니다.
따라서 실행 결과는 true를 반환합니다.
그런데, Java의 Equals() 요구 사항에 대해 이야기해 보겠습니다. 다음 사항이 있습니다.
대칭: x.equals(y)가 "true"를 반환하면 y.equals(x)도 "true"를 반환해야 합니다.
반사율: x.equals(x)는 "true"를 반환해야 합니다.
유추: x.equals(y)가 "true"를 반환하고 y.equals(z)가 "true"를 반환하면 z.equals(x)도 "true"를 반환해야 합니다.
일관성: x.equals(y)가 "true"를 반환하는 경우 x와 y의 내용이 변경되지 않는 한 x.equals(y)를 몇 번이나 반복하더라도 반환은 "true"가 됩니다. .
비어 있지 않은 x.equals(null)는 항상 "false"를 반환합니다. x.equals(x와 다른 유형의 개체)는 항상 "false"를 반환합니다.
이제 두 객체가 동일한지 확인하는 equals()의 역할을 검토해 보겠습니다. equals()를 다시 작성할 때 해당 기능을 변경하면 안 됩니다!


equals() 与 == 的区别是什么?
== : 它的作用是判断两个对象的地址是不是相等。即,判断两个对象是不是同一个对象。
equals() : 它的作用也是判断两个对象是否相等。但它一般有两种使用情况(前面第1部分已详细介绍过):
情况1,类没有覆盖equals()方法。则通过equals()比较该类的两个对象时,等价于通过“==”比较这两个对象。
情况2,类覆盖了equals()方法。一般,我们都覆盖equals()方法来两个对象的内容相等;若它们的内容相等,则返回true(即,认为这两个对象相等)。
下面,通过示例比较它们的区别。
代码如下:

import java.util.*;
import java.lang.Comparable;
 
/**
 * @desc equals()的测试程序。
 */
public class EqualsTest3{
 
 public static void main(String[] args) {
  // 新建2个相同内容的Person对象,
  // 再用equals比较它们是否相等
  Person p1 = new Person("eee", 100);
  Person p2 = new Person("eee", 100);
  System.out.printf("p1.equals(p2) : %s\n", p1.equals(p2));
  System.out.printf("p1==p2 : %s\n", p1==p2);
 }
 
 /**
  * @desc Person类。
  */
 private static class Person {
  int age;
  String name;
 
  public Person(String name, int age) {
   this.name = name;
   this.age = age;
  }
 
  public String toString() {
   return name + " - " +age;
  }
 
  /**
   * @desc 覆盖equals方法
   */
  @Override
  public boolean equals(Object obj){
   if(obj == null){
    return false;
   }
 
   //如果是同一个对象返回true,反之返回false
   if(this == obj){
    return true;
   }
 
   //判断是否类型相同
   if(this.getClass() != obj.getClass()){
    return false;
   }
 
   Person person = (Person)obj;
   return name.equals(person.name) && age==person.age;
  }
 }
}
로그인 후 복사

运行结果:

p1.equals(p2) : true
p1==p2 : false
로그인 후 복사

结果分析:
在EqualsTest3.java 中:
(1) p1.equals(p2)
这是判断p1和p2的内容是否相等。因为Person覆盖equals()方法,而这个equals()是用来判断p1和p2的内容是否相等,恰恰p1和p2的内容又相等;因此,返回true。
(2) p1==p2
这是判断p1和p2是否是同一个对象。由于它们是各自新建的两个Person对象;因此,返回false。

更多Java中判断对象是否相等的equals()方法使用教程相关文章请关注PHP中文网!

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

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

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

AI Clothes Remover

AI Clothes Remover

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

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

AI Hentai Generator

AI Hentai Generator

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

뜨거운 도구

메모장++7.3.1

메모장++7.3.1

사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전

SublimeText3 중국어 버전

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

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

드림위버 CS6

드림위버 CS6

시각적 웹 개발 도구

SublimeText3 Mac 버전

SublimeText3 Mac 버전

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

2025 년 상위 4 개의 JavaScript 프레임 워크 : React, Angular, Vue, Svelte 2025 년 상위 4 개의 JavaScript 프레임 워크 : React, Angular, Vue, Svelte Mar 07, 2025 pm 06:09 PM

이 기사는 2025 년에 상위 4 개의 JavaScript 프레임 워크 (React, Angular, Vue, Svelte)를 분석하여 성능, 확장 성 및 향후 전망을 비교합니다. 강력한 공동체와 생태계로 인해 모두 지배적이지만 상대적으로 대중적으로

Java의 클래스로드 메커니즘은 다른 클래스 로더 및 대표 모델을 포함하여 어떻게 작동합니까? Java의 클래스로드 메커니즘은 다른 클래스 로더 및 대표 모델을 포함하여 어떻게 작동합니까? Mar 17, 2025 pm 05:35 PM

Java의 클래스 로딩에는 부트 스트랩, 확장 및 응용 프로그램 클래스 로더가있는 계층 적 시스템을 사용하여 클래스로드, 링크 및 초기화 클래스가 포함됩니다. 학부모 위임 모델은 핵심 클래스가 먼저로드되어 사용자 정의 클래스 LOA에 영향을 미치도록합니다.

카페인 또는 구아바 캐시와 같은 라이브러리를 사용하여 자바 애플리케이션에서 다단계 캐싱을 구현하려면 어떻게해야합니까? 카페인 또는 구아바 캐시와 같은 라이브러리를 사용하여 자바 애플리케이션에서 다단계 캐싱을 구현하려면 어떻게해야합니까? Mar 17, 2025 pm 05:44 PM

이 기사는 카페인 및 구아바 캐시를 사용하여 자바에서 다단계 캐싱을 구현하여 응용 프로그램 성능을 향상시키는 것에 대해 설명합니다. 구성 및 퇴거 정책 관리 Best Pra와 함께 설정, 통합 및 성능 이점을 다룹니다.

Spring Boot Snakeyaml 2.0 CVE-2022-1471 문제 고정 Spring Boot Snakeyaml 2.0 CVE-2022-1471 문제 고정 Mar 07, 2025 pm 05:52 PM

이 기사는 원격 코드 실행을 허용하는 중요한 결함 인 Snakeyaml의 CVE-2022-1471 취약점을 다룹니다. Snakeyaml 1.33 이상으로 Spring Boot 응용 프로그램을 업그레이드하는 방법에 대해 자세히 설명합니다.

Node.js 20 : 주요 성능 향상 및 새로운 기능 Node.js 20 : 주요 성능 향상 및 새로운 기능 Mar 07, 2025 pm 06:12 PM

Node.js 20은 V8 엔진 개선, 특히 더 빠른 쓰레기 수집 및 I/O를 통해 성능을 크게 향상시킵니다. 새로운 기능에는 더 나은 webAssembly 지원 및 정제 디버깅 도구, 개발자 생산성 및 응용 속도 향상이 포함됩니다.

빙산 : 데이터 호수 테이블의 미래 빙산 : 데이터 호수 테이블의 미래 Mar 07, 2025 pm 06:31 PM

대규모 분석 데이터 세트를위한 오픈 테이블 형식 인 Iceberg는 데이터 호수 성능 및 확장 성을 향상시킵니다. 내부 메타 데이터 관리를 통한 Parquet/Orc의 한계를 해결하여 효율적인 스키마 진화, 시간 여행, 동시 W를 가능하게합니다.

Java에서 기능 프로그래밍 기술을 어떻게 구현할 수 있습니까? Java에서 기능 프로그래밍 기술을 어떻게 구현할 수 있습니까? Mar 11, 2025 pm 05:51 PM

이 기사는 Lambda 표현식, 스트림 API, 메소드 참조 및 선택 사항을 사용하여 기능 프로그래밍을 Java에 통합합니다. 간결함과 불변성을 통한 개선 된 코드 가독성 및 유지 관리 가능성과 같은 이점을 강조합니다.

오이의 단계간에 데이터를 공유하는 방법 오이의 단계간에 데이터를 공유하는 방법 Mar 07, 2025 pm 05:55 PM

이 기사는 오이 단계간에 데이터를 공유하는 방법, 시나리오 컨텍스트, 글로벌 변수, 인수 통과 및 데이터 구조를 비교합니다. 간결한 컨텍스트 사용, 설명을 포함하여 유지 관리에 대한 모범 사례를 강조합니다.

See all articles