Dagger 2 依赖关系图不完整

Dagger 2 dependency graph is not complete

我正在使用 Dagger 2 在我的新 Android 应用程序中注入依赖项。我想将一个 dao 注入到一个服务中。

模块:

@Module
public class DenkoStationModule {
    @Provides
    @Singleton
    public DenkoStationDao provideDenkoStationDao() {
        DaoMaster daoMaster = new DaoMaster(DenkoApplication.getDatabase());
        DaoSession daoSession = daoMaster.newSession();
        Log.d("dao", "station dao created");
        return daoSession.getDenkoStationDao();
    }

    @Provides
    @Singleton
    public DenkoStationService provideDenkoStationService() {
        Log.d("service", "station service created");
        return new DenkoStationService();
    }
}

服务:

public class DenkoStationService {

    private DenkoStationDao denkoStationDao;

    @Inject
    public void setDenkoStationDao(DenkoStationDao denkoStationDao) {
        this.denkoStationDao = denkoStationDao;
    }

    public List<DenkoStation> fetchAllDenkoStations() {
        Log.d("service", "loading all");
        return denkoStationDao.loadAll();
    }
}

组件:

@Singleton
@Component(modules = {DenkoStationModule.class})
public interface DenkoStationComponent {

    DenkoStationService provideDenkoStationService();
}

我从我的 activity 中使用它是这样的:

DenkoStationComponent denkoStationComponent = Dagger_DenkoStationComponent.builder().denkoStationModule(new DenkoStationModule()).build();
        DenkoStationService denkoStationService = denkoStationComponent.provideDenkoStationService();

        List<DenkoStation> denkoStations = denkoStationService.fetchAllDenkoStations();

看来 dao 没有被注入。

Caused by: java.lang.NullPointerException: Attempt to invoke virtual method 'java.util.List org.bitbucket.infovillafoundation.denko.dao.DenkoStationDao.loadAll()' on a null object reference
            at org.bitbucket.infovillafoundation.denko.service.DenkoStationService.fetchAllDenkoStations(DenkoStationService.java:23)
            at org.bitbucket.infovillafoundation.denko.activity.HomeActivity.onCreate(HomeActivity.java:29)
            at android.app.Activity.performCreate(Activity.java:5933)
            at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1105)
            at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2251)
            at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2360)
            at android.app.ActivityThread.access0(ActivityThread.java:144)
            at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1278)
            at android.os.Handler.dispatchMessage(Handler.java:102)
            at android.os.Looper.loop(Looper.java:135)
            at android.app.ActivityThread.main(ActivityThread.java:5221)
            at java.lang.reflect.Method.invoke(Native Method)
            at java.lang.reflect.Method.invoke(Method.java:372)
            at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:899)
            at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:694)

当您使用依赖项注入解决方案时,您希望框架也能注入依赖项的依赖项。是框架问题还是我做错了什么?

Dagger 2 使用注释处理器如果你想在你的对象中注入一些东西,你必须在使用实例之前调用负责注入的方法。有没有初始化Component的地方?如果没有任何地方你必须这样做。

如下所示在组件内部创建初始化器:

@Singleton
@Component(modules = {DenkoStationModule.class})
public interface DenkoStationComponent {


    public final static class Initializer {

        public static HierarchyViewerComponent init() {
            return Dagger_DenkoStationComponent.builder()
                    .denkoStationModule(new DenkoStationModule()) //it is not necessary because DenkoStationModule doesn't have params
                    .build();
        }

        void inject(DenkoStationService deviceInfoProvider);

    }
 }

之后你必须通过以下方法初始化你的组件:

DenkoStationComponent.Initializer.init();

Application 对象是执行此操作的好地方。你必须保留这个实例(它可以是静态的)。

现在您可以注入您的 Dao 对象,如下所示:

public class DenkoStationService {

   @Inject
   DenkoStationDao denkoStationDao;

    @Override
    public void onCreate() {
        super.onCreate();
        //lines below are responsible for inject your object

        //use line below if you keep component instance in some object 
        //ObjectWhichContainsComponent.methodToGetComponent().inject(this);
        //or
        DenkoStationComponent.Initializer.init().inject(this);

        //if you didn't initialise component before

    }

    public List<DenkoStation> fetchAllDenkoStations() {
        Log.d("service", "loading all");
        return denkoStationDao.loadAll();
    }
}

如有任何问题,请在评论中提问。

Here您可以找到更多信息。

您可以通过执行以下操作来解决您的问题。请注意,DenkoStationService 没有任何 @Inject 注释,因为服务是在提供者方法中创建的。 provideDenkoStationService() 的参数由 Dagger 传递。

@Module
public class DenkoStationModule {
  @Provides
  @Singleton
  public DenkoStationDao provideDenkoStationDao() {
    DaoMaster daoMaster = new DaoMaster(DenkoApplication.getDatabase());
    DaoSession daoSession = daoMaster.newSession();
    Log.d("dao", "station dao created");
    return daoSession.getDenkoStationDao();
  }

  @Provides
  @Singleton
  public DenkoStationService provideDenkoStationService(DenkoStationDao denkoStationDao) {
    Log.d("service", "station service created");
    return new DenkoStationService(denkoStationDao);
  }

}

public class DenkoStationService {
  private DenkoStationDao denkoStationDao;

  public DenkoStationService(DenkoStationDao denkoStationDao) {
    this.denkoStationDao = denkoStationDao;
  }

  public List<DenkoStation> fetchAllDenkoStations() {
    Log.d("service", "loading all");
    return denkoStationDao.loadAll();
  }
}

最近几天我一直在努力解决同样的问题,试图将我现有的 Dagger 1 代码迁移到 Dagger 2。

经过大量的反复试验,我偶然发现了一个解决方案 - 为您的@Provides 方法提供一个 MembersInjector<> 参数,并在创建对象后调用其 injectMembers() 方法。在您的情况下,您将如下所示修改 provideDenkoStationService()。

    @Provides
    @Singleton
    public DenkoStationService provideDenkoStationService(MembersInjector<DenkoStationService> injector) {
        Log.d("service", "station service created");
        DenkoStationService denkoStationService = new DenkoStationService();
        injector.injectMembers(denkoStationService);
        return denkoStationService;
    }

不幸的是,injectMembers() returns void 所以你不能将它压缩成一行,但一个方便的实用方法应该能够为你做到这一点。

我应该强调这是我自己的发现,我没有在任何地方看到这种行为的记录!不过它非常适合我。

我同意,基于我使用过的其他 DI 框架,以及我对 DI 的总体理解,这种功能应该免费提供或至少通过某种注释提供。特别是因为它确实在 Dagger 1 中起作用(同样,不是以最明显的方式,但解决方案更简单、更直观)

显然,使用另一个答案中建议的构造函数注入是另一种解决方案,但是对于具有多个依赖项的众多对象,需要编写大量样板代码,这通常会破坏 Dagger 的目的。另外,我无法让它注入我需要的 Provider 对象。