如何按类类型创建泛型类型安全的HashMap?

mem*_*und 6 java generics

我想创建一个HashMap将特定类类型映射到一个特定的新对象.

后来我想传递类类型并获取对该特定对象的引用.简单的例子:

Map<Class<?>, ?> values = new HashMap<>();

public <T> t get(Class<T> type) {
    return values.get(type);
}


//pet and car do not share any interface or parent class
class Pet;
class Car;

//error: not applicable for arguments
values.put(Pet.class, new Pet());
values.put(Car.class, new Car());
Run Code Online (Sandbox Code Playgroud)

用法:

values.get(Pet.class);
Run Code Online (Sandbox Code Playgroud)

如何相应地创建这样的通用散列图和查找函数?

Bhe*_*ung 5

我认为您正在寻找异构容器。在没有任何显式强制转换的情况下,没有简单的方法可以在 Java 中创建这样的容器。

我认为最简单的方法是将你的地图放在一个类中 -

public class Container {
    private Map<Class<?>, Object> values = new HashMap<>();

    public <T> T put(Class<T> klass, T obj) {
        return klass.cast(values.put(klass, obj));
    }

    public <T> T get(Class<T> klass) {
        return klass.cast(values.get(klass));
    }
}
Run Code Online (Sandbox Code Playgroud)

并按如下方式使用该类 -

Container c = new Container();
c.put(Pet.class, new Pet());
c.put(Car.class, new Car());
Pet p = c.get(Pet.class);
Run Code Online (Sandbox Code Playgroud)


ski*_*iwi 5

你需要存储某种类型的对象,如果你想Object在地图中放置其他东西,那么从这开始:

Map<Class<?>, Object> values = new HashMap<>();
Run Code Online (Sandbox Code Playgroud)

这必须以这种方式完成,因为?它不是具体的对象,而是Object对于地图存储的类型.

所以下面的代码片段可以正常工作,没有任何警告:

Map<Class<?>, Object> values = new HashMap<>();
values.put(Pet.class, new Pet());
values.put(Car.class, new Car());
Run Code Online (Sandbox Code Playgroud)

现在的诀窍是获取对象,我们按如下方式执行:

@SuppressWarnings("unchecked")
private <T> T get(Class<T> clazz) {
    return (T)values.get(clazz);
}
Run Code Online (Sandbox Code Playgroud)

现在,您的目标是确保映射包含在运行时不会出现错误的对.如果你放一个new Car()实例Car.class,那么你将会遇到错误.

所以下面的示例代码:

    values = new HashMap<>();
    values.put(Pet.class, new Pet());
    values.put(Car.class, new Car());

    System.out.println("get(Pet.class).getClass() = " + get(Pet.class).getClass());
    System.out.println("get(Car.class).getClass() = " + get(Car.class).getClass());
Run Code Online (Sandbox Code Playgroud)

将打印:

get(Pet.class).getClass() = class testproject8.Pet
get(Car.class).getClass() = class testproject8.Car
Run Code Online (Sandbox Code Playgroud)