java - Guice - 如何实现返回不同实现的工厂

标签 java guice factory

假设我有一个名为 Guice 服务的服务,这是它的构造函数

public GuiceService(IPayment payment) {
    this.payment = payment;
}

我的代码曾经使用枚举创建它

IPayment payment = new PaymentFactory.create(PaymentType.Cash);
NaiveService naiveService = new NaiveService(payment);

而且我必须在某个地方有一个工厂实现。像这样的东西

public IPayment create(PaymentType paymentType) {
    IPayment cardPayment = null;
    switch (paymentType) {
        case Cash:
            cardPayment = new CashPayment(100);
            break;
        case Card:
            cardPayment = new CardPayment(10, 100);
            break;
    }
    return cardPayment;

现在我想使用 Guice,我想我想使用 FactoryModuleBuilder。

  1. 如果我有多个 IPayment 实现,该怎么做。
    (例如 CardPayment、CashPayment)
    这适用于一个

    install(new FactoryModuleBuilder()
       .implement(IPayment.class, CashPayment.class)
       .build(IPaymentFactory.class));
    
  2. 如何实现构造函数?
    它还会得到IPayment吗?还是会获取 Guice 创建的 factoryImpl?

谢谢

最佳答案

您现有的实现是您可以获得的最佳实现。

为了清楚起见,让我们编写一个通用 IPaymentFactory:

public interface IPaymentFactory {
  IPayment create(/* ... */);
}

因此,IPaymentFactory 的实例定义了一个方法,该方法接受一些参数并返回一个 IPayment 的实例。您可以自己编写一个实现,显然您已经做到了,但是 Guice 的 FactoryModuleBuilder 会自动提供类似这样的接口(interface)实现。您永远不需要定义关于该类的任何其他内容:Guice 将为您连接构造函数,并将其绑定(bind)到 IPaymentFactory,这样您就可以注入(inject) IPaymentFactory 实例,使用您的参数调用 create(...),并获取 IPayment 实例。

看起来您想要的是一个采用枚举的工厂:

public interface IPaymentFactory {
  IPayment create(PaymentType paymentType);
}

...但是考虑到 CashPayment 接受一个任意参数,而 CardPayment 接受两个任意参数,并且考虑到它们之间的选择需要映射到任意 PaymentType 枚举,您还没有给 Guice 几乎 足够的信息来构建正确的对象。

Guice FactoryModuleBuilder 的设计更多是为了将构造函数参数与依赖项结合起来:

// Constructor:
@Inject public BitcoinPayment(
    @Assisted long value,         // varies by instance as a constructor parameter
    BitcoinService bitcoinService // passed-in dependency satisfied by Guice
    ) { /* ... */ }

// Factory interface:
public IBitcoinPaymentFactory {
  BitcoinPayment create(long value); // users don't need to know about dependencies!
}

// Factory binding...
install(new FactoryModuleBuilder().build(IBitcoinPaymentFactory.class));

// ...which lets Guice write the equivalent of:
public GeneratedBitcoinPaymentFactory implements IBitcoinPaymentFactory {
  @Inject Provider<BitcoinService> bitcoinServiceProvider;

  @Override public BitcoinPayment create(long value) {
    return new BitcoinPayment(value, bitcoinServiceProvider.get());
  }
}

一方面,工厂比您想象的要笨:它只是将参数与依赖项结合起来以获得一个完整的列表。另一方面,它很方便:您只需指定一次依赖项列表,其余的由 Guice 完成。

总结:FactoryModuleBuilder 不会解决您的问题,但它可以帮助您为 CashPayment 和 CardPayment 创建工厂,然后您可以将它们注入(inject)到您的手动 PaymentFactory 实现中(仍然需要存在)以某种形式)。

附言在您的示例中,这可能是用于演示的“玩具问题”,您可能不需要使用 Guice。对于需要依赖的服务对象,Guice 是一个很好的解决方案,但数据对象(如支付)或其他似乎不需要依赖的对象(如 GuiceService 或 NaiveService)可以直接使用构造函数来构造。一旦他们开始需要 Guice 注入(inject)的依赖项,就应该很容易让他们感知 Guice。

关于java - Guice - 如何实现返回不同实现的工厂,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25226912/

相关文章:

java - 在java中分别打印一个字符串中给出的字母和数字

java - 具有昂贵成员实例的 Guice Provider 是否应该使用 @Singleton 进行注释?

java - 泛型 hell : Can I construct a TypeLiteral<Set<T>> using generics?

java - 如何将依赖项注入(inject) Jackson 自定义反序列化器

java - 可以使用 Coldfusion 创建 JSR 286 portlet 吗?

java - 同一个包中的类彼此不通信

java - Hibernate和Mysql对象不刷新时间戳列

database - Laravel 工厂 - 创建或动态化

design-patterns - 工厂模式。什么时候使用工厂方法?

javascript - Node 简单对象工厂模块