当前位置: 首页 > 面试题库 >

Guice @Provides方法与提供程序类

淳于恺
2023-03-14
问题内容

我正在做一个很大的项目,有很多注入。当前,我们正在使用一个类,该类Provider为需要一次的每次注入实现,并且它们大多具有一个行get方法。

每当我需要一个新的提供程序时,创建一个新的类就变得很烦人。使用提供程序类比使用@Provides方法有什么好处,Module反之亦然?


问题答案:

据我所知,它们在大多数简单情况下是完全等效的。

/**
 * Class-style provider.
 * In module: bind(Foo.class).annotatedWith(Quux.class).toProvider(MyProvider.class);
 */
class MyProvider implements Provider<Foo> {
  @Inject Dep dep;  // All sorts of injection work, including constructor injection.

  @Override public Foo get() {
    return dep.provisionFoo("bar", "baz");
  }
}

/**
 * Method-style provider. configure() can be empty, but doesn't have to be.
 */
class MyModule extends AbstractModule {
  /** Name doesn't matter. Dep is injected automatically. */
  @Provides @Quux public Foo createFoo(Dep dep) {
    return dep.provisionFoo("bar", "baz");
  }

  @Override public void configure() { /* nothing needed in here */ }
}

无论哪种样式,即使键绑定到类或实例,Guice都可以让您注入FooProvider<Foo>get如果直接获取实例,Guice会自动调用,如果Provider<Foo>不存在则创建一个隐式实例。绑定注释在两种样式中均有效。

@Provides的主要优点是紧凑性,特别是与匿名内部Provider实现相比。但是请注意,在某些情况下,您可能希望使用Provider类:

  • 您可以创建自己的长期Provider实例(可能带有构造函数参数),并将键绑定到这些实例而不是类文字。

    bind(Foo.class).toProvider(new FooProvisioner("bar", "baz"));
    
  • 如果您使用的是与JSR 330(javax.inject)兼容的框架,则可以轻松绑定到javax.inject.Provider类或实例。com.google.inject.Provider扩展了该接口。

    bind(Foo.class).toProvider(SomeProviderThatDoesntKnowAboutGuice.class);
    
  • 您的提供者可能很复杂,无法纳入自己的类。根据您的测试结构,以这种方式测试提供程序可能会更容易。

  • 提供程序可以扩展抽象类。使用@Provides方法执行此操作可能并不容易或直观。

  • 您可以将多个密钥直接绑定到同一个提供程序。每个@Provides方法仅产生一个绑定,尽管您可以将其他键绑定到该 (此处为@Quux Foo),然后让Guice进行第二次查找。

  • 如果您想要(例如)在不使用Guice范围或绑定的情况下缓存或记忆实例,则提供程序很容易装饰或包装。

    bind(Foo.class).toProvider(new Cache(new FooProvisioner("bar", "baz")));
    

重要提示
:尽管这对于Guice无法创建的类来说是一个不错的策略,但是请记住,Guice可以自动Provider<T>为您创建的T并bind以任何方式(包括类名,键或实例)插入A。除非有您自己的实际逻辑,否则无需创建显式提供程序。



 类似资料:
  • 主要内容:Google Guice @Provides 完整示例,输出Google Guice 提供了一种使用 @Provides 注解创建复杂对象绑定的方法。 此方法是绑定模块的一部分,并提供要映射的复杂对象。请参阅下面的完整示例。 Google Guice @Provides 完整示例 创建一个名为 GuiceTester 的 Java 类。 GuiceTester.java 输出 编译并运行该文件,您将看到以下输出。

  • 我有一个Guice,它的构造函数接受注入的参数: 现在,我希望能够注入参数取决于我运行此参数的环境。在测试中,我想注入一个 MyConfiguration 对象,而在生产中,我想注入另一个对象。 我有两个MyConfiguration提供程序。MyConfigurationProvider读取外部配置文件并从那里获取配置。MyConfigurationTestProvider只是对所有设置进行硬编

  • 我想在我的片段(HomeFragment)中注入一个依赖项(HomeViewModel)。 我有一个类(HomeViewModelImpl)实现了该抽象(HomeViewModel),在这个类中,我当然覆盖了父级的方法。 抽象类(HomeViewModel)是从BaseViewModel扩展而来的抽象类。 BaseViewModel是一个普通的开放类,它从Android生命周期组件的ViewMod

  • 如果我创建一个提供者并将其绑定到一个类,就像这样 然后

  • 我的应用程序中有两个绑定类型的命名实例: 我有一个类,希望每个类使用一个实例。由于技术原因,此类不能直接注入实例,它必须向实例注入提供程序: 问题是,上面的FooPrime注入不是注入名为“prime”的实例,而是注入名为“prime”的提供者,这当然不是我想要的。 如何让Guice为名为“prime”的Foo实例注入一个提供程序?

  • 问题内容: 有人可以向我解释差异吗? Provisioner-在docker中安装,运行,拉出容器中的一项工作。 提供程序-是运行VM的东西。即VBox运行ubuntu OS映像。 Docker如何成为提供者?它直接运行一些docker镜像吗?如果我在Windows上,必须有一些boot2docker的隐藏用法,对不对?我什么时候使用每个? 问题答案: Docker Provisioner帮助准备