多模块应用中的 Hilt

Hilt 代码生成需要访问所有使用 Hilt 的 Gradle 模块。编译您的Application类的 Gradle 模块需要在其传递依赖项中包含所有 Hilt 模块和构造器注入类。

如果您的多模块项目由常规 Gradle 模块组成,则可以使用 Hilt,如使用 Hilt 进行依赖注入中所述。但是,包含功能模块的应用并非如此。

功能模块中的 Hilt

在功能模块中,模块通常相互依赖的方式是相反的。因此,Hilt 无法处理功能模块中的注释。您必须使用Dagger在您的功能模块中执行依赖注入。

您必须使用组件依赖项来解决功能模块中的此问题。请按照以下步骤操作

  1. app模块(或任何其他可由 Hilt 处理的模块)中声明一个@EntryPoint接口,其中包含功能模块所需的依赖项。
  2. 创建一个依赖于@EntryPoint接口的 Dagger 组件。
  3. 像往常一样在功能模块中使用 Dagger。

考虑使用 Hilt 进行依赖注入页面中的示例。假设您向项目添加了一个login功能模块。您使用名为LoginActivity的活动来实现登录功能。这意味着您只能从应用程序组件获取绑定。

对于此功能,您需要一个具有authInterceptor绑定的OkHttpClient

首先,创建一个安装在SingletonComponent中的@EntryPoint接口,其中包含login模块所需的绑定

Kotlin

// LoginModuleDependencies.kt - File in the app module.

@EntryPoint
@InstallIn(SingletonComponent::class)
interface LoginModuleDependencies {

  @AuthInterceptorOkHttpClient
  fun okHttpClient(): OkHttpClient
}

Java

// LoginModuleDependencies.java - File in the app module.

@EntryPoint
@InstallIn(SingletonComponent.class)
public interface LoginModuleDependencies {

  @AuthInterceptorOkHttpClient
  OkHttpClient okHttpClient();
}

要在LoginActivity中执行字段注入,请创建一个依赖于@EntryPoint接口的 Dagger 组件

Kotlin

// LoginComponent.kt - File in the login module.

@Component(dependencies = [LoginModuleDependencies::class])
interface LoginComponent {

  fun inject(activity: LoginActivity)

  @Component.Builder
  interface Builder {
    fun context(@BindsInstance context: Context): Builder
    fun appDependencies(loginModuleDependencies: LoginModuleDependencies): Builder
    fun build(): LoginComponent
  }
}

Java

// LoginComponent.java - File in the login module.

@Component(dependencies = LoginModuleDependencies.class)
public interface LoginComponent {

  void inject(LoginActivity loginActivity);

  @Component.Builder
  interface Builder {
    Builder context(@BindsInstance Context context);
    Builder appDependencies(LoginModuleDependencies loginModuleDependencies);
    LoginComponent build();
  }
}

完成这些步骤后,像往常一样在您的功能模块中使用 Dagger。例如,您可以使用SingletonComponent中的绑定作为类的依赖项

Kotlin

// LoginAnalyticsAdapter.kt - File in the login module.

class LoginAnalyticsAdapter @Inject constructor(
  @AuthInterceptorOkHttpClient okHttpClient: OkHttpClient
) { ... }

Java

// LoginAnalyticsAdapter.java - File in the login module.

public class LoginAnalyticsAdapter {

  private final OkHttpClient okHttpClient;

  @Inject
  LoginAnalyticsAdapter(
    @AuthInterceptorOkHttpClient OkHttpClient okHttpClient
  ) {
    this.okHttpClient = okHttpClient;
  }
  ...
}

要执行字段注入,请使用applicationContext创建一个 Dagger 组件的实例,以获取SingletonComponent依赖项

Kotlin

// LoginActivity.kt - File in the login module.

class LoginActivity : AppCompatActivity() {

  @Inject
  lateinit var loginAnalyticsAdapter: LoginAnalyticsAdapter

  override fun onCreate(savedInstanceState: Bundle?) {
    DaggerLoginComponent.builder()
        .context(this)
        .appDependencies(
          EntryPointAccessors.fromApplication(
            applicationContext,
            LoginModuleDependencies::class.java
          )
        )
        .build()
        .inject(this)

    super.onCreate(savedInstanceState)
    ...
  }
}

Java

// LoginActivity.java - File in the login module.

public class LoginActivity extends AppCompatActivity {

  @Inject
  LoginAnalyticsAdapter loginAnalyticsAdapter;

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    DaggerLoginComponent.builder()
        .context(this)
        .appDependencies(
          EntryPointAccessors.fromApplication(
            getApplicationContext(),
            LoginModuleDependencies.class
          )
        )
        .build()
        .inject(this);

    super.onCreate(savedInstanceState);
    ...
  }
}

有关功能模块中模块依赖项的更多上下文,请参阅使用功能模块的组件依赖项

有关 Android 上 Dagger 的更多信息,请参阅在 Android 应用中使用 Dagger