- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在尝试结合 Guice 的这 3 个功能:注入(inject)、多重绑定(bind)、泛型。我创建了一个生产项目的原型(prototype),所以这里是:
首先,这是泛型的一个小层次结构(在生产案例中有 N 个实体的层次结构):
public interface Type {
}
public class Type1 implements Type{
}
public class Type2 implements Type {
}
接下来,类 ToCreate1 和 ToCreate2 我想通过 Factory 创建。
基类:
public abstract class AbstractToCreate<T extends Type> {
public T type;
public Integer param;
public AbstractToCreate(T type, Integer param){
this.type = type;
this.param = param;
}
}
它的继承者:
public class ToCreate1 extends AbstractToCreate<Type1>{
@Inject
public ToCreate1(Type1 type, @Assisted Integer param) {
super(type, param);
}
}
public class ToCreate2 extends AbstractToCreate<Type2> {
@Inject
public ToCreate2(Type2 type, @Assisted Integer param) {
super(type, param);
}
}
然后,Factory 本身:
public interface Factory<T extends Type> {
AbstractToCreate<T> create(Integer param);
}
所以,现在我想注入(inject)一个映射,包含 Factory
因此,我使用配置方法创建了 Guice 的 AbstractModule:
protected void configure() {
install(new FactoryModuleBuilder()
.implement(new TypeLiteral<AbstractToCreate<Type1>>(){}, ToCreate1.class)
.build(new TypeLiteral<Factory<Type1>>(){}));
install(new FactoryModuleBuilder()
.implement(new TypeLiteral<AbstractToCreate<Type2>>(){}, ToCreate2.class)
.build(new TypeLiteral<Factory<Type2>>(){}));
MapBinder<String, Factory> mapBinder = MapBinder.newMapBinder(binder(), String.class, Factory.class);
mapBinder.addBinding("type1").to(new TypeLiteral<Factory<Type1>>(){});
mapBinder.addBinding("type2").to(new TypeLiteral<Factory<Type2>>(){});
}
所以,我注入(inject)它@Inject public Map<String, Factory> map;
一切正常:
Factory<Type1> factory1 = main.map.get("type1");
Factory<Type2> factory2 = main.map.get("type2");
AbstractToCreate<Type1> create1 = factory1.create(1);//create1 is ToCreate1 instance
AbstractToCreate<Type2> create2 = factory2.create(2);//create2 is ToCreate2 instance
正如我之前提到的,在我的生产系统中有更多的类型,所以 AbstractModule 变得太麻烦了。我试图避免重复代码并修改了 configure 方法:
@Override
protected void configure() {
this.<Type1>inst(ToCreate1.class);
this.<Type2>inst(ToCreate2.class);
}
private <V extends Type> void inst(Class<? extends AbstractToCreate<V>> clazz) {
install(new FactoryModuleBuilder()
.implement(new TypeLiteral<AbstractToCreate<V>>(){}, clazz)
.build(new TypeLiteral<Factory<V>>(){}));
}
而且它不起作用! Guice 说:
1) ru.test.genericassistedinject.AbstractToCreate<V> cannot be used as a key; It is not fully specified.
怎么了?
最佳答案
这里的问题是类型删除。特别是这段代码:
private <V extends Type> void inst(Class<? extends AbstractToCreate<V>> clazz) {
install(new FactoryModuleBuilder()
.implement(new TypeLiteral<AbstractToCreate<V>>(){}, clazz)
.build(new TypeLiteral<Factory<V>>(){}));
}
无法工作,因为它依赖于类型参数 V
帮助做出运行时决定(使用什么绑定(bind)),但类型参数 V
没有运行时表示,因此它的值永远不会直接影响运行时。另一种思考方式:Java 不能“读取”泛型中类型参数的值; new TypeLiteral<Factory<V>>(){}
无论 V
是什么,总是相同的值在调用者中实例化。
当您遇到与删除相关的问题时,通常会出现这种情况,诀窍是添加一个代表您想要的类型的运行时值。在这种情况下,这尤其棘手,因为您要做的是将类型参数的值表示为更大的类型。
有几种方法可以获取表示静态类型的运行时值。 TypeToken
是一个并且Class
是另一个,但它们都不允许您用参数表示类型,然后以编程方式填充该值。幸运的是,Google Guava 包含另一种表示形式,com.google.common.reflect.TypeToken
,这对我们有用。 TypeToken
s 可以表示具有变量的类型,并支持以编程方式“填充”具有具体表示的变量,例如:
new TypeToken<List<V>>() {}.where(new TypeParameter<V>() {}, Integer.class)
代表类型List<Integer>
在运行时。
使用 TypeToken
我们可以像这样构建我们的类型:
private <V extends Type> void inst(Class<? extends AbstractToCreate<V>> clazz, Class<V> binding) {
TypeToken<AbstractToCreate<V>> implementationType = new TypeToken<AbstractToCreate<V>>() {}
.where(new TypeParameter<V>() {}, binding);
TypeToken<Factory<V>> factoryType = new TypeToken<Factory<V>>() {}
.where(new TypeParameter<V>() {}, binding);
@SuppressWarnings("unchecked") // The type returned by TypeToken::getType is always the type it represents
Key<AbstractToCreate<V>> key = (Key<AbstractToCreate<V>>) Key.get(implementationType.getType());
@SuppressWarnings("unchecked") // as above
Key<Factory<V>> factoryKey = (Key<Factory<V>>) Key.get(factoryType.getType());
install(
new FactoryModuleBuilder()
.implement(key, clazz)
.build(factoryKey));
}
现在我们可以调用inst
与:
inst(ToCreate1.class, Type1.class);
inst(ToCreate2.class, Type2.class);
一切都会按预期工作。
不过,这是非常奇特的东西,理解它取决于对类型的编译时和运行时表示之间的区别有很好的理解。如果是我,如果你只希望使用一次或两次,我不会这样做,因为混淆负担非常高;我只会在它是图书馆或其他东西的一部分时这样做,并且您可以为每个来电者节省一些工作。
关于java - guice 辅助注入(inject) + 多绑定(bind) + 泛型,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49769830/
我有很多 TabularResultSet 类的带注释的命名实例,我想将它们传递给多个类并传递给静态最终属性。我该如何注入(inject)?我认为它不喜欢该属性的最终和静态性质。必须在属性声明中直接调
我是 Guice 的新手,我想知道我能走多远。 我有一个接口(interface)UserInfo具有多个实现类 GoogleUserInfo , FacebookUserInfo , Twitter
我的应用程序中绑定(bind)了两个类型的命名实例: bind(Foo.class).toProvider(FooProvider.class); bind(Foo.class).annotatedW
我有一个 Guice 模块,它有一个 @Provides 方法,它接受 2 个参数并返回接口(interface)的实现: public class **ClientModule** extends
请解释在以下使用工厂的场景中使用 Google-Guice 自动连接的正确方法。我正在使用 XML 工厂进行 XSLT 处理。 StringWriter strWriter = new StringW
我有一个 Guice Module提供 List使用 @Provides - 带注释的方法。 class TestModule() : Module { override fun configur
为什么Guice 3.0针对错误的配置组件(例如,缺少@Inject)抛出此异常,而不是格式化消息? Exception in thread "main" com.google.inject.inte
我有以下供应商: public class GuiceResourceProvider implements Provider { @Inject private Configur
使用 Guice-servlet,我们可以轻松地进行 servlet 映射,如下所示: filter(*.jsp).through(MyFilter.class) 但是,谁能告诉我如何将过滤器映射到
我正在尝试向我的项目添加 Swagger 。我们的设置与示例项目略有不同。我们使用 guice 和 guice-servlet 来注入(inject)并启动我们的 JerseyServletModul
假设我有一个 Guice 模块 ProdModule,我想依赖其他 GuiceModule、ProdDbModule 和 ProdPubSubModule。我将如何实现 ProdModule 的 co
我正在考虑在我的应用程序中使用 Guice for DI,我应该能够在运行时交换实现。下面提供了一个示例来说明要求: class ValidationEngine { public void v
我想设置 Guice 绑定(bind),所以我用 Java 创建了一个完美运行的模块: public class CrashLoggerModule extends AbstractModule {
所以我有一个模块将接口(interface)与实现类绑定(bind)。 bind(ILocalStore.class).to(LocalStore.class); 此绑定(bind)是否还会注入(in
guice 4.0 是否向后兼容 3.x?无法从发行说明或常见问题解答中弄清楚... 如果没有,是否有兼容性问题列表? 最佳答案 我没有任何官方来源但根据我的经验,没有任何兼容性问题。 我使用了几个
我正在尝试使用 Guice,并且我来自 Spring。 我想知道 @Inject 是否相当于 Spring 中的 @Autowired 以及我是否可以在 Web 应用程序中使用它,就像在 Spring
我是 Guice DI 的新手。我想弄清楚我的情况。 简单来说,有没有通过Guice @annotations来替代MapBinder的? 我的场景: Interface A{} Class A1 i
我的项目正在使用 Guice作为负责为大型对象图(主要是单例)提供依赖项(服务类)的 IOC 容器。有时,如果在构造过程中依赖项失败,并且许多对象都需要此依赖项,则失败将一遍又一遍地发生,并将异常添加
我有一个类 (CustomConnectionProvider),它将由第三方库 (hibernate) 使用 class.forName().newInstance() 实例化。我需要注入(inje
删除 guice servlet 后,我需要进行一些清理。使用 guice servlet 时是否可以 Hook 到 servlet 破坏?我需要使用喷油器进行清理工作。 我可以覆盖 contex
我是一名优秀的程序员,十分优秀!