在Java中编写用于继承的equals()方法
我在做一个继承问题,我已经完成了所有的事情,除了布尔部分。我的代码将被编译,但是当比较名称、生日和ssn时,它只会显示为false 例如,我输入:在Java中编写用于继承的equals()方法,java,inheritance,boolean,Java,Inheritance,Boolean,我在做一个继承问题,我已经完成了所有的事情,除了布尔部分。我的代码将被编译,但是当比较名称、生日和ssn时,它只会显示为false 例如,我输入: 吉姆 2001年6月30日 123456789 吉姆 2001年6月30日 123456789 输出将为false public class Person { private String name; private Date birthday; private int ssn; public Person(S
- 吉姆
- 2001年6月30日
- 123456789
- 吉姆
- 2001年6月30日
- 123456789
public class Person
{
private String name;
private Date birthday;
private int ssn;
public Person(String name, Date birthday, int ssn)
{
this.name = name;
this.birthday = birthday;
this.ssn = ssn;
}
public String getName()
{
return name;
}
public Date getBirthday()
{
return birthday;
}
public int getSSN()
{
return ssn;
}
/* I already called a dates toString() method for birthday */
public String toString()
{
return name + " has birthday " + birthday + " and SSN " + ssn;
}
public boolean equals(Object otherObj)
{
if(otherObj == null)
return false;
else if(otherObj.getClass() != this.getClass())
return false;
else
{
Person otherP = (Person)otherObj;
if(otherP.name.equals(this.name) &&
otherP.birthday == this.birthday &&
otherP.ssn == this.ssn)
return true;
else
return false;
}
}
}
这里没有继承权。事实上,在从超类继承时编写equals方法会引起关于类设计的完全不同的讨论(更喜欢组合而不是继承) 在这种情况下,我认为您需要遵循经验法则,即始终使用equals方法对所有字段进行逻辑比较检查用于标识比较,很少用于逻辑相等(例如枚举) 另一点需要注意的是,您的类将在使用哈希的数据结构中表现不好
Joshua Bloch的《高效Java》第3章详细介绍了这个主题,我认为问题在于:
if(otherP.name.equals(this.name) &&
otherP.birthday == this.birthday &&
otherP.ssn == this.ssn)
上面的代码是检查两个Person
对象是否相等的主要代码。您说,当您提供两个生日、名称和SSN完全相同的对象时,就会出现问题。我不确定您的输入代码是如何创建对象的,但从我所知,它可能会为每个生日创建一个新的日期
=
运算符和等于方法之间有一个重要区别。任何类型的变量都包含一个值。对于基本数据类型(例如,int
),变量的值就是它实际存储的值。但是,对于对象变量(例如,String
,Date
),该值实际上是对对象的引用。如果两个对象变量引用堆上的同一对象,则它们将具有相同的值;如果它们引用堆上的不同对象,则它们将具有不同的值
==
运算符只检查变量的值,而不检查其他值。当它与原语一起使用时,它会检查变量中存储的值是否相同,这是正确的,因为这些值是直接存储的。当它与对象变量一起使用时,它会检查这两个变量是否引用堆上的同一个对象。由于变量的值实际上是引用,这是检查两个值或引用是否相同时的结果
另一方面,equals
方法实际上(应该)进一步检查两个对象是否被视为相等,因为它们的状态相同。例如,它可以检查两个Date
对象是否表示相同的日期。Java无法神奇地判断两个不同的对象是否相等——类必须实现equals
方法本身。(否则,它会求助于默认实现,只检查引用。)
在上面的代码示例中,您在原语ssn
上正确地使用=
操作符(因为它是原语),在对象名称上正确地使用equals
方法(因为每个Person
中的引用可能不同,但表示相同的字符串)。但是,您在对象变量生日
上使用=
运算符,它只检查正在比较的两个人
对象是否具有相同的日期
对象。很可能,因为调用代码只是在每个输入上创建一个新的Date
对象,所以检查失败,因为这两个Person
对象有两个不同的生日对象。只需将其更改为equals
方法调用,您的代码就会正常工作。为什么要使用=
来比较生日?通常,您希望使用.equals
来比较对象。继承在哪里?总是,当覆盖equals
时,您还应该覆盖hashCode
。否则,许多类,如HashSet
(使用hashCode
)将无法按预期工作。如果您使用IDE,您应该寻找一个选项来生成equals
和hashCode
。生成的代码可能比您自己编写的代码工作得更好。equals
的原因应该用于String
对任何对象都有效=
只比较身份,而equals
经常比较内容/属性。顺便说一句,我建议你不要将过时很久的Date
类用作生日。现代的java.time.LocalDate
既能更好地使用,又能更精确地满足您对一天中没有小时的日期的要求。