views:

724

answers:

1

I am trying to migrate a small project, replacing some factories with Guice (it is my first Guice trial). However, I am stuck when trying to inject generics. I managed to extract a small toy example with two classes and a module:

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);
  }

}

When I run this example, all I got is:

Exception in thread "main" com.google.inject.CreationException: Guice creation errors:

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)

I tried looking for the error message, but without finding any useful hints. Further on the Guice FAQ I stumble upon a question about how to inject generics. I tried to add the following binding in the configure method:

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

But without success (same error message).

Can someone explain me the error message and provide me some tips ? Thanks.

+3  A: 

I think the specific issue you're seeing is probably because of the bind(Console.class) statement... it should use a TypeLiteral as well. Or, you could just bind neither of those and JIT bindings will take care of it for you since both of the types involved here are concrete classes.

Additionally, you should retrieve the Console with:

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

Edit: You don't need to bind to an instance just because you're using a TypeLiteral. You can just do:

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

Of course, like I said above you could just skip that in this case and retrieve the Console from the injector using a Key based on the TypeLiteral and the binding would be implicit.

ColinD
Thanks. I am not sure how to bind a TypeLitteral to Console<Double> because I cannot make an instance of Console without a StringOutput instance. Can you detail ?
paradigmatic
See my edit... you don't need to use toInstance().
ColinD
I just tried. It works only if I define both type litterals AND y use the Key to get the instance. So I am not able to make the binding explicit.
paradigmatic
I'm not sure what you mean by not being able to make the binding explicit. If you put the bind statement for each in your module, it's explicit. Also, I should note that in general you should only have ONE call to injector.getInstance() in an application, which should retrieve some root application object for you to start. Everything else should be worked out by @Inject annotated constructors and such. So in practice, you shouldn't generally need to use the injector.getInstance(Key.get(new TypeLiteral...)) thing.
ColinD
Sorry I meant implicit.
paradigmatic