gpt4 book ai didi

Java泛型通配符问题

转载 作者:搜寻专家 更新时间:2023-10-30 19:46:54 24 4
gpt4 key购买 nike

在使用 Google Guava 出色的 Multimap 时,我遇到了一些泛型问题。我有一个这样定义的类型处理程序

public interface Handler<T extends Serializable> {
void handle(T t);
}

在另一个类中,我定义了一个将字符串映射到处理程序集合的多重映射。

private Multimap<String, Handler<? extends Serializable>> multimap = 
ArrayListMultimap.create();

现在,当我尝试使用多图进行操作时,我遇到了编译器错误。我的第一次尝试是这样的:

public <T extends Serializable> void doStuff1(String s, T t)  {
Collection<Handler<T>> collection = multimap.get(s);
for (Handler<T> handler : collection) {
handler.handle(t);
}
}

导致以下错误。

Type mismatch: cannot convert from Collection<Handler<? extends Serializable>>
to Collection<Handler<T>>

后来,我试着这样写

public void doStuff2(String s, Serializable serializable)  {
Collection<Handler<? extends Serializable>> collection = multimap.get(s);
for (Handler<? extends Serializable> handler : collection) {
handler.handle(serializable);
}
}

不幸的是,它也失败了:

The method handle(capture#1-of ? extends Serializable) in the type
Handler<capture#1-of ? extends Serializable> is not applicable for the arguments
(Serializable)

如有任何帮助,我们将不胜感激。谢谢。

更新:

我设法解决这个问题的唯一方法是抑制编译器警告。给定以下处理程序:

public interface Handler<T extends Event> {
void handle(T t);

Class<T> getType();
}

我可以这样写事件总线。

public class EventBus {

private Multimap<Class<?>, Handler<?>> multimap = ArrayListMultimap.create();

public <T extends Event> void subscribe(Handler<T> handler) {
multimap.put(handler.getType(), handler);
}

@SuppressWarnings({ "rawtypes", "unchecked" })
public void publish(Event event) {
Collection<Handler<?>> collection = multimap.get(event.getClass());
for (Handler handler : collection) {
handler.handle(event);
}
}
}

我想没有办法用更少甚至没有 @SuppressWarnings 来处理这个问题?

最佳答案

问题是类型可能不同:

private Multimap<String, Handler<? extends Serializable>> multimap = 
ArrayListMultimap.create();

不允许您向多图添加任何内容,因为您不知道 ? 是什么实际上代表。例如,你可以有一个 Multimap<String, Handler<String>>并尝试添加 Integer因为两者都实现了 Serializable .

编辑:其实上面这段话有点错误。您应该能够向多重映射添加处理程序,但由于处理程序的类型参数未知,您将无法使用这些处理程序,请参见下文。

在你的doStuff1定义具体参数的方法T这可能是完全不同的东西。因此编译器无法确定这个赋值是否正确:Collection<Handler<T>> collection = multimap.get(s); (T 真的是您从多重映射中获得的处理程序的类型吗?- 编译器不知道)。

您的第二种方法确实可以正确分配,但是 handle()方法将不起作用,因为您传入了 Serializable这可能是任何东西( StringInteger ,其他东西)并且编译器仍然不知道处理程序的类型是否匹配(想象它是一个 Handler<Number> 并且你将 String 传递给 doStuff2 )。

您有多种方法可以解决这个问题,每种方法都有其自身的缺点:

  1. 只需使用 Multimap<String, Handler<Serializable>> ,这将允许您传递任何 Serializable反对处理程序
  2. 使用具体类型,例如Multimap<String, Handler<String>> ,这将限制您仅使用字符串处理程序
  3. 在运行时获取处理程序的类型参数并进行转换,如果不正确可能会出错

关于Java泛型通配符问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6000116/

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