3

我有一组 15-20 个类,它们依赖于一个实现具有绑定类型参数的泛型接口的类。界面如下所示:

interface Handler<F extends Foo, T extends Foo> {
    T handleFoo(String methodName, F myFoo);
}

还有一个 Handler 的实现,它使用反射来处理 Foo 的所有特殊情况,因为它们都是“相似但不同”的。

class ConcreteHandler<Foo, Foo> implements Handler<Foo, Foo> {
    T handleFoo(String methodName, F foo) {
        // do your thing
    }
}

我希望能够在其协作者中以类型安全的方式使用此类,例如:

@Inject ACollaborator(Handler<Bar, Baz> barHandler) {...}

whereBarBazboth 延伸FooHandler<Bar, Baz>不幸的是,当我在我的模块中尝试这个时,Guice 抱怨说没有绑定:

bind(new TypeLiteral<Handler<? extends Foo, ? extends Foo>>() {}).to(ConcreteHandler.class);

我还尝试了以下方法,这会导致编译器错误,因为ConcreteHandler它不是的子类Handler<Bar, Baz>

bind(new TypeLiteral<Handler<Bar, Baz>>() {}).to(ConcreteHandler.class);

我有很多这样的 Collaborator 对象,以至于我并不热衷于为每种类型组合实现 Handler 并为它们放入 20 个单独的绑定语句。

我如何完成我正在尝试使用 Guice 的功能,即将单个具体实现绑定到具有不同泛型参数的单个接口?

4

1 回答 1

2

您可以通过创建表示绑定目标的 TypeLiteral 来设法做到这一点,可能涉及到原始类型。

您将在此问题中看到如何执行此操作的示例: Reconstructing generic types at runtime with Guice via Types and TypeLiterals

基本上你需要的是构建你想要绑定的每个 Handler<> 的 Key 的东西。

然后你可以这样说:

bindHandler(Bar.class, Baz.class);

这将:

  1. 创建一个TypeLiteral<ConcreteFoo<Bar, Baz>>
  2. 创建一个TypeLiteral<Foo<Bar, Baz>>
  3. 将后者绑定到前者

另外我认为在你的例子中 ConcreteHandler 应该实现 Handler[?]

于 2013-09-20T13:19:56.907 回答