Ошибка конфигурации Guice: реализация не была связана

Я пытаюсь сделать DI с работой Guice, делая (как мне кажется) именно то, что в руководство.

Я не могу объяснить проблему, потому что я ее не понимаю - все кажется очень логичным и должно работать. Но это не так. Итак, я могу только прикрепить код и стек:

public class Runner {

    public static void main(String[] args) {
        Injector injector = Guice.createInjector(new TestModule());
        //next line throws the exception
        JMeterComponent jMeterComponent = 
             injector.getInstance(JMeterComponent.class);
        ....
    }
}

Как вы можете видеть, я пытаюсь создать объект класса JMeterComponent. Конструктор (как вы увидите ниже) принимает 3 аргумента: все они должны быть также созданы IoC и введены.

И здесь TestModule с конфигурацией этих трех аргументов:

public class TestModule extends AbstractModule {

    @Override
    protected void configure() {
        bind(Callable.class).annotatedWith(Names.named("JMeter"))
                  .to(JMeterTask.class);      
        bind(Processor.class).annotatedWith(Names.named("JMeter"))
                  .to(JMeterResultsProcessor.class);
        bind(Renderer.class).annotatedWith(Names.named("JMeter"))
                  .to(JMeterResultsWikiRenderer.class);
    }
}

Теперь посмотрим на конкретные реализации - JMeterTask, JMeterResultsProcessor и JMeterResultsWikiRenderer (для простоты все они имеют поддельные тела):

public class JMeterTask implements Callable<JMeterRawResults> {

    public JMeterRawResults call() throws Exception {
        return new JMeterRawResults();
    }
}

public class JMeterResultsProcessor implements 
                   Processor<JMeterRawResults, JMeterResults> {

    public JMeterResults process(JMeterRawResults raw) {
        return new JMeterResults();
    }
}

public class JMeterResultsWikiRenderer implements Renderer<JMeterResults> {

    public Map<String, String> render(JMeterResults jMeterResults) {
        Map<String, String> results = Maps.newHashMap();
        ...
        return results;
    }
}

И теперь давайте посмотрим на класс JMeterComponent, конструкция экземпляра которого является целью всего связанного с DI материала:

public class JMeterComponent extends AbstractComponent<String, String> {

    @Inject
    public JMeterComponent(@Named("JMeter") Callable<String> task, 
                           @Named("JMeter")Processor<String, String> processor, 
                           @Named("JMeter")Renderer<String> renderer) {
        super(task, processor, renderer);
    }
}

И вот stacktrace:

Exception in thread "main" com.google.inject.ConfigurationException: Guice configuration errors:

1) No implementation for stat.domain.Processor<java.lang.String, java.lang.String> annotated with @com.google.inject.name.Named(value=JMeter) was bound.
  while locating stat.domain.Processor<java.lang.String, java.lang.String> annotated with @com.google.inject.name.Named(value=JMeter)
    for parameter 1 at stat.components.jmeter.JMeterComponent.<init>(JMeterComponent.java:18)
  while locating cstat.components.jmeter.JMeterComponent

2) No implementation for stat.domain.Renderer<java.lang.String> annotated with @com.google.inject.name.Named(value=JMeter) was bound.
  while locating stat.domain.Renderer<java.lang.String> annotated with @com.google.inject.name.Named(value=JMeter)
    for parameter 2 at stat.components.jmeter.JMeterComponent.<init>(JMeterComponent.java:18)
  while locating stat.components.jmeter.JMeterComponent

3) No implementation for java.util.concurrent.Callable<java.lang.String> annotated with @com.google.inject.name.Named(value=JMeter) was bound.
  while locating java.util.concurrent.Callable<java.lang.String> annotated with @com.google.inject.name.Named(value=JMeter)
    for parameter 0 at stat.components.jmeter.JMeterComponent.<init>(JMeterComponent.java:18)
  while locating stat.components.jmeter.JMeterComponent

Некоторые дополнительные факты:

  • Я использую guice-2.0 (с featured метки)
  • Больше никаких комментариев из пакета com.google.inject в любом другом классе кода
  • Интерфейсы Processor и Renderer помещаются в один модуль, а их классы jmeter -implementations (JMeterResultsProcessor и др.) и JMeterComponent помещаются в другой модуль.

Это почти все, что можно сказать об этом.

Извините за такой длинный пост и спасибо за ваше терпение, чтобы прочитать его до конца.

Любые идеи о том, почему произошли ошибки и как их исправить?

Ответ 1

Есть несколько проблем, которые я вижу здесь.

Во-первых, Callable и Callable<String> отличаются. Если вы хотите ввести a Callable<String> (или Processor<String, String> и т.д.) В Guice, вам нужно связать что-то с Callable<String>, а не Callable.

Во-вторых, вы привязываете Callable к JMeterTask, который реализует Callable<JMeterRawResults>, но вы вставляете Callable<String> в конструктор JMeterComponent (то же самое для Processor и Renderer). Я предполагаю, что JMeterComponent должен иметь Callable<JMeterRawResults> и т.д., Введенный.

Во всяком случае, вам нужно делать общие привязки с помощью TypeLiteral:

bind(new TypeLiteral<Callable<JMeterRawResults>>(){})
    .annotatedWith(Names.named("JMeter"))
    .to(JMeterTask.class);