gpt4 book ai didi

Java 泛型 : How to specify a Class type for a generic typed class?

转载 作者:塔克拉玛干 更新时间:2023-11-03 03:21:36 26 4
gpt4 key购买 nike

我有一个 POJO 指定为:MyClass<U> , 其中U是泛型类型参数。我正在尝试编写一个接受类引用的实用方法 Class<T>并填充 Map<String, T> 类型的 map (接受 map 填充)。

这个方法是这样实现的:

static void populateMap(Map<String, T> map, Class<T> type) {

...

// Parses into the specified type and returns an object of that type.
T obj = parse(..., type);
map.put (key, obj);
...

return map;
}

这编译得很好。在我的来电者中,我尝试用任何 MyClass 填充 map 实例(不考虑类型)作为值。因此我使用以下代码:

// Loses type information
Map<String, MyClass<?>> m = new HashMap<>();
populateMap(m, MyClass.class);

这不会编译。编译错误:

The method populate(Map<String,T>, Class<T>) in the type ... is not applicable for the arguments (Map<String,MyClass<?>>, Class<MyClass>)

我该如何解决这个问题?

最佳答案

在这种情况下,对 Class<MyClass<?>> 进行未经检查的转换应该是安全的:

// This is okay because we're switching to a type with an unbounded wildcard -
// the behaviors of Class.newInstance and Class.cast are still safe.
@SuppressWarnings("unchecked")
Class<MyClass<?>> classWithNarrowedType =
(Class<MyClass<?>>)(Class<?>)MyClass.class;
populateMap(m, classWithNarrowedType);

这是一个笨拙的解决方案,特别是如果您有许多这样的调用站点,但无法回避这样一个事实,即类文字使用原始类型进行参数化,使其用作参数化类型的工厂,如 MyClass<T>天生尴尬。

一个可能更清洁的解决方案将解耦 populateMap来自类文字的使用:

interface Parser<T> {

T parse();
}

static void populateMap(Map<String, T> map, Parser<T> parser) { ... }

...

Map<String, MyClass<?>> m = new HashMap<>();
Parser<MyClass<?>> myClassParser = new Parser<MyClass<?>>() {
@Override
public MyClass<?> parse() {
return parse(..., MyClass.class);
}
};
populateMap(m, myClassParser);

顺便说一句,我建议使用更灵活的签名(有关更多信息,请参阅 What is PECS (Producer Extends Consumer Super)?):

static void populateMap(Map<String, ? super T> map, Parser<T> parser)

关于Java 泛型 : How to specify a Class type for a generic typed class?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18940376/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com