Warning: file_get_contents(/data/phpspider/zhask/data//catemap/9/java/305.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
在Java中编写用于继承的equals()方法_Java_Inheritance_Boolean - Fatal编程技术网

在Java中编写用于继承的equals()方法

在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

我在做一个继承问题,我已经完成了所有的事情,除了布尔部分。我的代码将被编译,但是当比较名称、生日和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(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
既能更好地使用,又能更精确地满足您对一天中没有小时的日期的要求。