我可以使用 Dagger 的某种辅助注入吗?

Can I use some kind of assisted Inject with Dagger?

使用 Google Guice 或 Gin 我可以指定参数不受依赖注入框架控制:

class SomeEditor {


  @Inject
  public SomeEditor(SomeClassA a, @Assisted("stage") SomeClassB b) {
  }

}

在创建 SomeEditor 的实例时指定辅助参数 stage

SomeClassA 的实例取自对象图,SomeClassB 的实例取自运行时的调用者。

在 Dagger 中有类似的方法吗?

因为工厂是一种单独类型的样板,需要优化(see mailing list discussion here), Dagger leaves it to a sister project, AutoFactory. This provides the "assisted injection" functionality Guice offers via FactoryModuleBuilder,但有一些额外的好处:

  • 您可以继续将 AutoFactory 与 Guice 或 Dagger 或任何其他 JSR-330 依赖项注入框架一起使用,因此即使您在它们之间切换也可以继续使用 AutoFactory。
  • 因为 AutoFactory 生成代码,所以您不需要编写接口来表示构造函数:AutoFactory 会编写一个全新的类型供您编译。 (如果您愿意,或者如果您要从 Guice 迁移,您也可以指定要实现的接口。)
  • 因为所有类型检查都发生在编译时,它会生成普通的旧 Java,它不会因反射而导致任何缓慢,并且适用于调试器和优化器。这使得 Auto 库对 Android 开发特别有用。

示例,从 AutoFactory 的自述文件中提取,它将在 @Inject 注释的构造函数中生成 SomeClassFactoryprovidedDepA,在 create 中生成 depB方法:

@AutoFactory
final class SomeClass {
  private final String providedDepA;
  private final String depB;

  SomeClass(@Provided @AQualifier String providedDepA, String depB) {
    this.providedDepA = providedDepA;
    this.depB = depB;
  }

  // …
}

是的,请检查这个 Square 项目:square/AssistedInject

目前它不在 1.0 中。他们等到 Dagger 将引入 public API 来自动注册那些生成的 Module 类 - 请参阅 this issue. With that you won't have to reference them in your Dagger code as in this example from README:

@AssistedModule
@Module(includes = AssistedInject_PresenterModule.class)
abstract class PresenterModule {}

就像@xsveda一样,我也写了一个关于这个的答案in this other question,我也会在这里复制它。


今天,对于 Dagger 的辅助注入,您可能想使用 AssistedInject. I wrote about it in this blogpost,但我将在此处添加一个完整的示例以使事情变得更容易。

首先你需要的是依赖项:

compileOnly 'com.squareup.inject:assisted-inject-annotations-dagger2:0.4.0'
kapt 'com.squareup.inject:assisted-inject-processor-dagger2:0.4.0'

那么它是这样的:

class ImageDownloader @AssistedInject constructor(
  private val httpClient: HttpClient,
  private val executorService: ExecutorService,
  @Assisted private val imageUrl: URL,
  @Assisted private val callback: ImageCallback
) {

  @AssistedInject.Factory
  interface Factory {
    fun create(imageUrl: URL, callback: ImageCallback): ImageDownloader
  }
}

首先,我们没有用 @Inject 注释构造函数,而是用 @AssistedInject 注释它。然后我们注释必须通过工厂的参数,这与 AutoFactory 期望的相反。最后,我们需要一个用 @AssistedInject.Factory 注释的内部工厂接口,它有一个接收辅助参数的方法和 returns 我们感兴趣的实例。

不幸的是,我们这里还有一个额外的步骤:

@AssistedModule
@Module(includes = [AssistedInject_AssistedInjectModule::class])
interface AssistedInjectModule

我们不一定需要专用模块,即使这是一个有效选项。但是我们也可以在组件中已经安装的另一个模块中使用这些注释。这里的好处是我们只需要做一次,之后任何工厂都会自动成为图表的一部分。

有了它,您基本上可以像往常一样注入工厂并请求您的对象。