Dagger singleton每次创建新实例

时间:2017-11-24 08:17:52

标签: java android dependency-injection dagger-2

我有一个模块如下。

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

1 个答案:

答案 0 :(得分:3)

  1. 在您的模块中,您有一个提供SoundsRepository实例的方法 - 好
  2. 在您的AppComponent中,您缺少:

    SoundsRepository soundsRepository();
    
  3. 在扩展Application / MultidexApplication的Global中,您可以创建DaggerAppComponent - 好

  4. 在您的其他活动/片段/服务中,只需致电:

    Global application = (Global) getApplication();
    SoundsRepository sr = application.getComponent().soundsRepository()
    
  5. 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(...);
    }
    

    会导致您的问题