9

为了好玩,我正在尝试实现一个“MultiMap”集合,就像 Apache Commons 库中已经存在的那样。我的“remove(K key, V value)”方法遇到了一个有趣的错误。编译器说存在名称冲突 - 它与“类型 Map 的删除(对象,对象)”具有相同的擦除。但是java.util.Map接口中没有定义这样的方法!只有一个“删除(对象)”方法 - 一个参数,而不是我的两个参数版本。更有趣的是,如果您通过将我的“remove(K key, V value)”替换为“remove(Object key, Object value)”来手动删除类型信息,它编译得很好。谁能解释这种现象?

我正在运行 Java 8,以防万一。

import java.util.AbstractMap;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.Map;
import java.util.Set;

public class MultiMap<K, V> extends AbstractMap<K, Collection<V>>
{
    private Map<K, Collection<V>> map;

    public MultiMap()
    {
        super();
        map = new HashMap<>();
    }

    //Fine
    public void clear(K key)
    {
        get(key).clear();
    }

    //Fine
    public boolean add(K key, V value)
    {
        if(!containsKey(key))
            put(key, new ArrayList<>());
        return get(key).add(value);
    }

    //KABOOM!!
    //"Name clash: The method remove(K, V) of type MultiMap<K,V> has the same erasure as remove(Object, Object) of type Map<K,V> but does not override it"
    public boolean remove(K key, V value)
    {
        if(!containsKey(key))
            return false;
        return get(key).remove(value);
    }

    @Override public Collection<V> put(K key, Collection<V> values)
    {
        return map.put(key, values);
    }

    @Override public Set<java.util.Map.Entry<K, Collection<V>>> entrySet()
    {
        return map.entrySet();
    }
}
4

2 回答 2

15

但是java.util.Map接口中没有定义这样的方法!

接口中有一个Map#remove(Object, Object)方法Map;它是在 Java 8 中添加的。因此出现了错误。

于 2014-05-21T14:18:16.393 回答
1

这是将 java 泛型类型转换为字节码时的类型擦除过程。

在运行时default boolean remove(Object key, Object value)Map 中的(此方法是 Java 8 中的新方法)与 MultiMap 类中的方法相同public boolean remove(K key, V value)

编译器看到了这一点,因此显示了错误:

名称冲突:MultiMap 类型的方法 remove(K,V) 与 AbstractMap 类型的 remove(Object, Object) 具有相同的擦除,但不会覆盖它。

看到这个:http ://docs.oracle.com/javase/tutorial/java/generics/genTypes.html

于 2014-05-21T19:41:34.413 回答