Я пытаюсь сделать 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
помещаются в другой модуль.
Это почти все, что можно сказать об этом.
Извините за такой длинный пост и спасибо за ваше терпение, чтобы прочитать его до конца.
Любые идеи о том, почему произошли ошибки и как их исправить?