Как заставить Multibinder работать с FactoryModuleBuilder и Generics, используя Guice?
Я прочитал /questions/25562894/google-guice-multibinding-obobscheniya-assistedinject, но, похоже, он не отвечает моему конкретному сценарию. Вот что у меня есть:
Интерфейсы для использования с FactoryModuleBuilder
public interface IFilterFactory<T extends IFilter> {
T create(FilterConfig config);
}
public interface IFilterOneFactory extends IFilterFactory<FilterOne> {}
public interface IFilterTwoFactory extends IFilterFactory<FilterTwo> {}
Модуль Мой Guice:
@Override
protected void configure() {
install(new FactoryModuleBuilder().build(IFilterOneFactory.class));
install(new FactoryModuleBuilder().build(IFilterTwoFactory.class));
Multibinder<IFilterFactory<?>> filterBinder = Multibinder.newSetBinder(binder(), new TypeLiteral<IFilterFactory<?>>() {});
filterBinder.addBinding().to(IFilterOneFactory.class);
filterBinder.addBinding().to(IFilterTwoFactory.class);
}
Мой основной фильтр, который должен объединять все остальные фильтры:
public class MainFilterFactory {
@Inject
MainFilterFactory(Set<IFilterFactory<?>> factories) {
this.factories = factories;
}
public MainFilter get(String groupKey) {
FilterConfig fc = new FilterConfig(groupKey);
IFilter[] filters = this.factories.stream().map(f -> f.create(fc)).toArray(IFilter[]::new);
return new MainFilter(Arrays.asList(filters));
}
}
Это на самом деле не работает. И я получаю следующие трассировки стека:
[WARNING]
java.lang.reflect.InvocationTargetException
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:497)
at org.codehaus.mojo.exec.ExecJavaMojo$1.run(ExecJavaMojo.java:297)
at java.lang.Thread.run(Thread.java:745)
[WARNING]
com.google.inject.internal.util.$ComputationException: java.lang.ArrayIndexOutOfBoundsException: 19640
at com.google.inject.internal.util.$MapMaker$StrategyImpl.compute(MapMaker.java:553)
at com.google.inject.internal.util.$MapMaker$StrategyImpl.compute(MapMaker.java:419)
at com.google.inject.internal.util.$CustomConcurrentHashMap$ComputingImpl.get(CustomConcurrentHashMap.java:2041)
at com.google.inject.internal.util.$StackTraceElements.forType(StackTraceElements.java:70)
at com.google.inject.internal.Errors.formatSource(Errors.java:687)
at com.google.inject.internal.Errors.format(Errors.java:555)
at com.google.inject.CreationException.getMessage(CreationException.java:48)
at java.lang.Throwable.getLocalizedMessage(Throwable.java:391)
at java.lang.Throwable.toString(Throwable.java:480)
at java.lang.String.valueOf(String.java:2982)
at java.lang.StringBuilder.append(StringBuilder.java:131)
at java.lang.Throwable.printEnclosedStackTrace(Throwable.java:695)
at java.lang.Throwable.printStackTrace(Throwable.java:667)
at java.lang.Throwable.printStackTrace(Throwable.java:643)
at org.apache.maven.cli.PrintStreamLogger.warn(PrintStreamLogger.java:131)
at org.apache.maven.monitor.logging.DefaultLog.warn(DefaultLog.java:93)
at org.codehaus.mojo.exec.ExecJavaMojo$IsolatedThreadGroup.uncaughtException(ExecJavaMojo.java:386)
at org.codehaus.mojo.exec.ExecJavaMojo$1.run(ExecJavaMojo.java:309)
at java.lang.Thread.run(Thread.java:745)
Caused by: java.lang.ArrayIndexOutOfBoundsException: 19640
at com.google.inject.internal.asm.$ClassReader.<init>(Unknown Source)
at com.google.inject.internal.asm.$ClassReader.<init>(Unknown Source)
at com.google.inject.internal.asm.$ClassReader.<init>(Unknown Source)
at com.google.inject.internal.util.$LineNumbers.<init>(LineNumbers.java:62)
at com.google.inject.internal.util.$StackTraceElements$1.apply(StackTraceElements.java:36)
at com.google.inject.internal.util.$StackTraceElements$1.apply(StackTraceElements.java:33)
at com.google.inject.internal.util.$MapMaker$StrategyImpl.compute(MapMaker.java:549)
... 18 more
Я что-то здесь не так делаю?
1 ответ
На самом деле это проблема совместимости Guice между Guice 3.0 и Java 8. См.: https://github.com/google/guice/issues/757
Мне не хватало некоторых @Assisted
а также @Inject
в конструкторах и Guice не удалось сообщить об ошибке, потому что некоторые лямбды использовались в этих классах.
Комментирование лямбд позволило мне увидеть ошибки и исправить их. Затем я раскомментировал их, и все, кажется, работает как задумано.