我有一个模块如下。
@Module
public class AppModule {
private final Application app;
public AppModule(Application app) {
this.app = app;
}
@Provides
@Architecture.ApplicationContext
Context provideContext() {
return app;
}
@Provides //scope is not necessary for parameters stored within the module
public Context context() {
return provideContext();
}
@Singleton
@Provides
Application provideApp() {
return app;
}
@Singleton
@Provides
SoundsRepository provideSoundsRepository(Context context, SoundsDAO soundsDAO) {
return new SoundsRepository(context, soundsDAO);
}
}
这样的组件。
@Singleton
@Component(modules = AppModule.class)
public interface AppComponent {
void inject(Global global);
void inject(MainActivity mainActivity);
@Architecture.ApplicationContext
Context getContext();
Application getApplication();
void inject(PostView postView);
void inject(MediaPlayerService mediaPlayerService);
}
在活动,片段或服务中,我这样做
@Inject
SoundsRepository soundsRepository;
@Override
protected void onCreate(...) {
//....
((Global) getApplication()).getComponent().inject(this);
}
在SoundsRepository中
@Singleton
public class SoundsRepository {
@Inject
public SoundsRepository(Context context, SoundsDAO soundsDAO) {
this.context = context;
this.soundsDAO = soundsDAO;
System.out.println(TAG + "INIT");
}
// ....
}
所以,现在,每当我开始访问注入了 SoundsRepository 的活动或服务时,我得到一个新实例,我的意思是,“SoundsRepository”的构造函数再次触发。
我做错了什么?
编辑:在应用程序类中注入
public class Global extends MultiDexApplication {
protected AppComponent appComponent;
private boolean calledAlready = false;
@Override
public void onCreate() {
super.onCreate();
//if (LeakCanary.isInAnalyzerProcess(this)) return;
//LeakCanary.install(this);
initFirebasePersistance();
appComponent = DaggerAppComponent.builder().appModule(new AppModule(this)).build();
appComponent.inject(this);
FrescoUtil.init(getApplicationContext());
}
public AppComponent getComponent() {
return appComponent;
}
}
答案 0 :(得分:3)
在您的AppComponent中,您缺少:
SoundsRepository soundsRepository();
在扩展Application / MultidexApplication的Global
中,您可以创建DaggerAppComponent - 好
在您的其他活动/片段/服务中,只需致电:
Global application = (Global) getApplication();
SoundsRepository sr = application.getComponent().soundsRepository()
Android保证您所有其他活动/服务只有一个Application(Global)类实例(有点像单身)。
请将您的组件放在该应用程序类中,并在需要课程时致电: (YourApplication) getApplication().getComponent().yourSingleInstanceSomething();
我为您创建并测试了示例代码:https://github.com/zakrzak/StackDaggerTest
Dagger的@Singleton只是一个范围,并不保证返回一个类的单个实例。
根据我的理解,如果你:
void inject(PostView postView);
你告诉Dagger,只要你提出请求,就可以在你的PostView中使用AppModule中的@Provided进行注释:
@Inject
SoundsRepository soundsRepository;
然后dagger只调用@provided方法,在你的情况下返回一个新的SoundRepository实例:
@Singleton
@Provides
SoundsRepository provideSoundsRepository(Context ........) {
return new SoundsRepository(...);
}
会导致您的问题