Can Guice可以根据参数自动创建不同类的实例吗?

时间:2013-07-16 06:23:13

标签: java dependency-injection inversion-of-control guice

标准对象工厂可能如下所示:

interface I { ... }
class A implements I { ... }
class B implements I { ... }

class IFactory {
    I getI(int i) {
        switch (i) {
        case 1: return new A();
        default: return new B();
        }
    }
}

是否可以设置绑定以便为我完成切换,即我所做的只是调用getInstance或者注入?我正在寻找辅助注射,但这似乎是不同的主题:https://code.google.com/p/google-guice/wiki/AssistedInject

1 个答案:

答案 0 :(得分:14)

听起来您正在寻找MapBinder,这是Multibindings功能的一部分。请注意,您仍需要输入某种IFactory或其他工厂界面,因为getInstance不会像getI那样采用参数,您仍然会需要建立从整数到类实现的映射某处

MapBinder式

class IModule extends AbstractModule {
  @Override public void configure() {
    MapBinder<Integer, I> myBinder =
        MapBinder.newMapBinder(binder(), Integer.class, I.class);
    myBinder.addBinding(1).to(A.class);
    // Add more here.
  }
}

// You can even split the MapBinding across Modules, if you'd like.
class SomeOtherModule extends AbstractModule {
  @Override public void configure() {
    // MapBinder.newMapBinder does not complain about duplicate bindings
    // as long as the keys are different.
    MapBinder<Integer, I> myBinder =
        MapBinder.newMapBinder(binder(), Integer.class, I.class);
    myBinder.addBinding(3).to(C.class);
    myBinder.addBinding(4).to(D.class);
  }
}

配置了这些模块的注入器将提供一个注入Map<Integer, I>,它具有绑定的所有内容的实例;这里是一个三条目映射,从1到完全注入A实例,从3到C实例,从4到D实例。这实际上是对您的切换示例的改进,该示例使用new关键字,因此未将任何依赖项注入AB

对于没有创建如此多浪费实例的更好选项,请注入MapBinder自动提供的Map<Integer, Provider<I>>。像这样使用它:

class YourConsumer {
  @Inject Map<Integer, Provider<I>> iMap;

  public void yourMethod(int iIndex) {
    // get an I implementor
    I i = iMap.get(iIndex).get();
    // ...
  }
}

要提供“默认”实现(和不透明的界面),但是,您需要在MapBinder地图之上实现自己的短包装:

class IFactory {
  @Inject Map<Integer, Provider<I>> iMap;
  @Inject Provider<B> defaultI; // Bound automatically for every Guice key

  I getI(int i) {
    return iMap.containsKey(i) ? iMap.get(i).get() : defaultI.get();
  }
}

更简单,工厂风格

如果以上内容看起来有点过分,请记住您可以inject an Injector并从密钥到实现创建本地Map。 (你也可以像我在这里一样使用ImmutableMap。)

class IFactory {
  @Inject Injector injector; // This is a bad idea, except for times like this
  @Inject Provider<B> defaultI;
  static final ImmutableMap<Integer, Class<? extends I>> map = ImmutableMap.of(
      1, A.class,
      3, C.class,
      4, D.class);

  I getI(int i) {
    return map.containsKey(i)
        ? injector.getInstance(map.get(i))
        : defaultI.get();
  }
}