29

我正在尝试迁移一个小项目,用 Guice 替换一些工厂(这是我的第一次 Guice 试用版)。但是,我在尝试注入泛型时被卡住了。我设法提取了一个带有两个类和一个模块的小玩具示例:

import com.google.inject.Inject;

public class Console<T> {
  private final StringOutput<T> out;
  @Inject
  public Console(StringOutput<T> out) {
    this.out = out;
  }
  public void print(T t) {
    System.out.println(out.converter(t));
  }
}

public class StringOutput<T> {
  public String converter(T t) {
    return t.toString();
  }
}

import com.google.inject.AbstractModule;
import com.google.inject.Guice;
import com.google.inject.Injector;
import com.google.inject.TypeLiteral;


public class MyModule extends AbstractModule {

  @Override
  protected void configure() {
    bind(StringOutput.class);
    bind(Console.class);
  }

  public static void main(String[] args) {
    Injector injector = Guice.createInjector( new MyModule() );
    StringOutput<Integer> out = injector.getInstance(StringOutput.class);
    System.out.println( out.converter(12) );
    Console<Double> cons = injector.getInstance(Console.class);
    cons.print(123.0);
  }

}

当我运行这个例子时,我得到的是:

线程“主”com.google.inject.CreationException 中的异常:Guice 创建错误:

1) playground.StringOutput<T> cannot be used as a key; It is not fully specified.
  at playground.MyModule.configure(MyModule.java:15)

1 error
    at com.google.inject.internal.Errors.throwCreationExceptionIfErrorsExist(Errors.java:354)
    at com.google.inject.InjectorBuilder.initializeStatically(InjectorBuilder.java:152)
    at com.google.inject.InjectorBuilder.build(InjectorBuilder.java:105)
    at com.google.inject.Guice.createInjector(Guice.java:92)

我尝试查找错误消息,但没有找到任何有用的提示。在 Guice FAQ 上,我偶然发现了一个关于如何注入泛型的问题。我尝试在configure方法中添加以下绑定:

bind(new TypeLiteral<StringOutput<Double>>() {}).toInstance(new StringOutput<Double>());

但没有成功(同样的错误信息)。

有人可以向我解释错误消息并提供一些提示吗?谢谢。

4

1 回答 1

28

我认为您看到的具体问题可能是由于该bind(Console.class)声明。它也应该使用 a TypeLiteral。或者,您可以不绑定其中任何一个,而 JIT 绑定将为您处理它,因为这里涉及的两种类型都是具体的类。

此外,您应该检索Consolewith:

Console<Double> cons = 
   injector.getInstance(Key.get(new TypeLiteral<Console<Double>>(){}));

编辑:您不需要仅仅因为使用TypeLiteral. 你可以这样做:

bind(new TypeLiteral<Console<Double>>(){});

当然,就像我上面所说的那样,在这种情况下,您可以跳过它,并使用基于 的Consolea 从注入器中检索 ,并且绑定将是隐式的。KeyTypeLiteral

于 2010-04-05T21:23:04.043 回答