如何为泛型类型实现equals?

时间:2017-06-01 11:59:22

标签: java generics type-erasure

假设我有一个像这样的通用容器类型:

public final class Container<T> {

    public final T t;

    public Container(final T t) {
        this.t = t;
    }
}

我想实现等于这样的传递:

final Container<Object> a = new Container<>("Hello");
final Container<String> b = new Container<>("Hello");

assertNotEquals(a, b);

实例ab应该不同,因为它们的类型参数T不同。

然而,由于擦除,这很棘手。例如,此实现不正确:

@Override
public boolean equals(final Object obj) {
    if (this == obj) {
        return true;
    }
    if (obj != null && obj instanceof Container<?>) {
        final Container<?> other = (Container<?>)obj;
        return Objects.equals(this.t, other.t);
    }
    return false;
}

我希望我需要为T存储某种令牌。

如何为泛型类型实现equals?

This没有回答这个问题。

1 个答案:

答案 0 :(得分:4)

你可以稍微修改一下Container类并添加这个字段:

public final Class<T> ct;

然后用等于覆盖

System.out.println(a.equals(b));

将返回false,因为equals方法会检查Class<String> vs Class<Object>

class Container<T> {

    public final T t;
    public final Class<T> ct;

    public Container(final T t, Class<T> ct) {
        this.t = t;
        this.ct = ct;
    }

    @Override
    public int hashCode() {
        final int prime = 31;
        int result = 1;
        result = (prime * result) + ((ct == null) ? 0 : ct.hashCode());
        result = (prime * result) + ((t == null) ? 0 : t.hashCode());
        return result;
    }

    @Override
    public boolean equals(Object obj) {
        if (this == obj)
            return true;
        if (obj == null)
            return false;
        if (getClass() != obj.getClass())
            return false;
        Container other = (Container) obj;
        if (ct == null) {
            if (other.ct != null)
                return false;
        } else if (!ct.equals(other.ct))
            return false;
        if (t == null) {
            if (other.t != null)
                return false;
        } else if (!t.equals(other.t))
            return false;
        return true;
    }

}