Dagger 2依赖图未完成

时间:2015-03-27 04:22:03

标签: android dagger dagger-2

我正在使用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();
}

我在我的活动中使用它:

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.access$800(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)

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

3 个答案:

答案 0 :(得分:1)

Dagger 2使用注释处理器如果要在对象中注入某些内容,则必须在使用实例之前调用负责注入的方法。您是否有任何初始化Component的地方?如果没有任何地方你必须这样做。

在Component中创建Initializer,如下所示:

@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();

应用程序对象是一个很好的地方。你必须保持这个例子(它可以是静态的)。

现在你可以注入你的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您可以找到更多信息。

答案 1 :(得分:1)

您可以通过以下方式解决问题。请注意,DenkoStationService没有任何@Inject注释,因为正在提供程序方法中创建服务。 Dagger传递provideDenkoStationService()的参数。

@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();
  }
}

答案 2 :(得分:0)

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

经过大量的反复试验,我偶然发现了一个解决方案 - 提供一个MembersInjector&lt;&gt; @Provides方法的参数,并在创建对象后调用其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()返回void,所以你不能将它压缩成一行,但是一个方便的实用方法应该能够为你做到这一点。

我应该强调这是我自己的发现,我没有看到这种行为记录在任何地方!它虽然对我来说很完美。

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

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