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 对象。
我正在使用 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 对象。