覆盖“等于”方法:如何弄清楚参数的类型?

2022-09-02 09:35:42

我正在尝试重写参数化类的方法。equals

@Override
public boolean equals(Object obj) {
    if (this == obj)
        return true;
    if (obj == null)
        return false;
    if (!(obj instanceof Tuple))
        return false;

    Tuple<E> other = (Tuple<E>) obj; //unchecked cast
    if (!a0.equals(other.a0) && !a0.equals(other.a1)) {
        return false;
    }
    if (!a1.equals(other.a1) && !a1.equals(other.a0)) {
        return false;
    }

    return true;
}

如何确保对象与 对象相同?<E>otherthis


答案 1

您可以通过保留对类型的引用来执行此操作。但是,在我看来,相等性测试应该是关于对象表示的值,而不是值所表达的具体类型Class<E>

例如,集合 API 就是一个典型的示例。 返回。虽然它们具有完全不同的类型,但它们表示相同的值,即“空集合”。new ArrayList<String>().equals(new LinkedList<Object>())true

就个人而言,表示相同数据(例如)的两个s是否应该不相等,因为一个是类型,而另一个是类型?Tuple("a", "b")Tuple<String>Tuple<Object>


答案 2

由于擦除,你不能。您能做的最好的事情就是在元组类中存储您计划在“java.lang.Class”字段成员中保留元组的类型。然后,您可以比较这些字段,以确保元组类具有相同的类型。

另请参阅此主题:Java 中的 C++ Pair<L,R> 等效项是什么?

如果您发布有关您的课程的更多信息,这将有所帮助。我在想,未经检查的强制转换和你等同的字段数意味着它应该是元组<E,F>不是吗?

编辑:这是一个我经常使用的有用的配对类(如果需要,您可以调整元组类)。请注意,与其他人的建议类似,此类只是让包含的成员决定相等性问题。您的用例应该确定相等性是否真的基于所包含成员的类型。

/**
 * Adapted from http://forums.sun.com/thread.jspa?threadID=5132045
 * 
 * 
 * @author Tim Harsch
 *
 * @param <L>
 * @param <R>
 */
public class Pair<L, R> {

    private final L left;
    private final R right;

    public R getRight() {
        return right;
    } // end getter

    public L getLeft() {
        return left;
    } // end getter

    public Pair(final L left, final R right) {
        this.left = left;
        this.right = right;
    } // end constructor

    public static <A, B> Pair<A, B> create(A left, B right) {
        return new Pair<A, B>(left, right);
    } // end factory method

    @Override
    public final boolean equals(Object o) {
        if (!(o instanceof Pair<?,?>))
            return false;

        final Pair<?, ?> other = (Pair<?, ?>) o;
        return equal(getLeft(), other.getLeft()) && equal(getRight(), other.getRight());
    } // end method

    public static final boolean equal(Object o1, Object o2) {
        if (o1 == null) {
            return o2 == null;
        }
        return o1.equals(o2);
    } // end method

    @Override
    public int hashCode() {
        int hLeft = getLeft() == null ? 0 : getLeft().hashCode();
        int hRight = getRight() == null ? 0 : getRight().hashCode();

        return hLeft + (37 * hRight);
    } // end method

    @Override
    public String toString() {
        StringBuilder sb = new StringBuilder();
        sb.append('<');
        if( left == null ) {
            sb.append("null");
        } else {
            sb.append(left.toString());
        } // end if
        sb.append(',');
        if( right == null ) {
            sb.append("null");
        } else {
            sb.append(right.toString());
        } // end if
        sb.append('>');
        return sb.toString();
    } // end method
} // end class

推荐