Передать параметр в конструктор с помощью Guice - PullRequest
31 голосов
/ 11 февраля 2012

У меня есть фабрика, как показано ниже,

public final class Application {

    private static IFoo foo;

    public static IFoo getFoo(String bar)
    {
        // i need to inject bar to the constructor of Foo
        // obvious i have to do something, not sure what
        Injector injector = Guice.createInjector();
        logger = injector.getInstance(Foo.class);
        return logger;              
    }

}

Это определение Foo:

class Foo
{
   Foo(String bar)
   {

   }

}

ОК.Я не уверен, как передать этот параметр конструктору Foo с помощью Guice?

Есть идеи?

Ответы [ 4 ]

52 голосов
/ 11 февраля 2012

Все ответы на «Guice Constructor Parameter» в некотором роде кажутся неполными. Вот полное решение, включая использование:

interface FooInterface{
  String getFooName();
}

// Аннотировать конструктор и вспомогательные параметры в классе реализации

class Foo implements FooInterface {
   String bar;

   @Inject
   Foo(@Assisted String bar)
   {
      this.bar = bar;
   }

   // return the final name
   getFooName(){
     return this.bar;
   }

}

// Создать фабричный интерфейс с методом create (), который принимает только вспомогательные параметры.

// Интерфейс FooFactory не имеет явного класса реализации (Guice Magic)

interface FooFactory{
   Foo create(String bar);
}

// Привязать эту фабрику к провайдеру, созданному AssistedInject

binderModule implements Module{

 void configure(Binder binder) {
   binder.install(new FactoryModuleBuilder()
         .implement(FooInterface.class, Foo.class)
         .build(FooFactory.class));
 }
}

// Теперь используйте его:

class FooAction{

   @Inject private FooFactory fooFactory;

   doFoo(){
      // Send bar details through the Factory, not the "injector" 
      Foo f = fooFactory.create("This foo is named bar. How lovely!");
      f.getFooName(); // "This foo is named bar. How lovely!"
   }
}

Здесь много подсказок: https://google.github.io/guice/api-docs/latest/javadoc/index.html?com/google/inject/assistedinject/FactoryModuleBuilder.html

8 голосов
/ 11 февраля 2012

Что вы, вероятно, ищете, так это использование фабрики Guice. Особенно легко с функциональностью AssistedInject, но у них есть ручной пример в верхней части страницы . Суть в том, что касается ручного примера, это то, что вы получаете фабрику нестатическим методом getFoo, который передает все необходимые параметры и строит объект оттуда.

Это не будет работать напрямую, если у вас есть перехват метода в Foo, но это будет работать во многих других случаях.

Чтобы использовать AssistedInject, который для меня имеет несколько более чистую семантику и означает меньшую ручную разводку, вам потребуется расширение guice-assistedinject в classpath, а затем при создании Foo (ну, FooImpl, мы должны использовать интерфейсы):

@Inject
public FooImpl(@Assisted String bar)
{
    this.baz = bar;
}

Затем вы создаете FooFactory интерфейс:

public interface FooFactory {
    public Foo create(String bar);
}

Тогда в вашем модуле guice:

install(new FactoryModuleBuilder()
    .implement(Foo.class, FooImpl.class)
    .build(FooFactory.class));

Вы можете проверить javadoc для FactoryModuleBuilder для примеров с более сложными фабриками.

7 голосов
/ 25 мая 2017

Я знаю, что это старая ветка, но сегодня я сам решил эту проблему. Мне нужны только два или максимум три разных экземпляра 'Foo', и я действительно не хотел писать весь болгарный код Factory. Немного погуглив, я нашел этот Стаббизис - блог Тони Я бы предложил это решение, которое идеально, если вы точно знаете, какие экземпляры вам нужны.

В модуле Guice:

    bind(Foo.class).annotatedWith(Names.named("firstFoo")).toProvider(new Provider<Foo>() {
        @Override
        public Foo get() {
            return new FooImpl("topic A");
        }
    });
    bind(Foo.class).annotatedWith(Names.named("secondFoo")).toProvider(new Provider<Foo>() {
        @Override
        public Foo get() {
            return new FooImpl("topic B");
        }
    });

Или в Java 8:

    bind(Foo.class).annotatedWith(Names.named("firstFoo")).toProvider(() -> new FooImpl("first"));
    bind(Foo.class).annotatedWith(Names.named("secondFoo")).toProvider(() -> new FooImpl("second"));

А в конструкторе вашего сервиса, где вам нужны экземпляры Foo:

@Inject
public MyService (
    @Named("firstFoo") Foo firstFoo,
    @Named("secondFoo") Foo secondFoo) {
}

И Фу в моем случае:

public class FooImpl implements Foo {

    private String name;

    public FooImpl(String name) {
        this.name = name;
    }

    @Override
    public String getName() {
        return name;
    }
}

Надеюсь, это кому-нибудь поможет.

1 голос
/ 11 февраля 2012

Если этот класс является фабрикой, то это должен быть управляемый Guice объект, имеющий нестатический метод getFoo, а метод getFoo будет просто использовать

new Foo(bar)

Не каждый экземпляр должен быть созданGuice.

Также см. AssistedInject , чтобы избежать создания этой фабрики самостоятельно и позволить Guice создать ее для вас.

...