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中文网!