如何使用dagger2注入没有任何活动或片段的java类

ant*_*009 1 android dependency-injection dagger-2

Android Studio 2.2.2
Run Code Online (Sandbox Code Playgroud)

我有一个NewsListModelImp类,它是MVP中的模型.

我想将改装服务注入模型中.但是,因为NewsListModelImp不包含对我无法调用的上下文或活动的任何引用getApplication().如果您在活动或片段中,您将采取以下措施.我不想在构造函数中传递任何上下文或活动,NewsListModeImp因为它必须来自演示者,我想避免任何android的东西.

public class NewsListModelImp implements NewsListModelContract {
    @Inject
    NYTimesSearchService mNYTimesSearchService;

    public NewsListModelImp() {
        ((NYTimesSearchApplication)getApplication()).getAppComponent().inject(this);
    }
}
Run Code Online (Sandbox Code Playgroud)

我的应用程序类

public class NYTimesSearchApplication extends Application {
    private AppComponent mAppComponent;

    public void onCreate() {
        super.onCreate();

        /* Setup dependency injection */
        createAppComponent();
    }

    private void createAppComponent() {
        mAppComponent = DaggerAppComponent
                .builder()
                .retrofitModule(new RetrofitModule())
                .build();
    }

    public AppComponent getAppComponent() {
        return mAppComponent;
    }
}
Run Code Online (Sandbox Code Playgroud)

我提供的模块

@Module
public class RetrofitModule {
    private Retrofit retrofit() {
        return new Retrofit
                .Builder()
                .baseUrl(Constants.BASE_URL)
                .addConverterFactory(GsonConverterFactory.create())
                .build();
    }

    @Provides
    @Singleton
    public NYTimesSearchService providesNYTimesSearch() {
        return retrofit().create(NYTimesSearchService.class);
    }
}
Run Code Online (Sandbox Code Playgroud)

我的Appcomponent

@Singleton
@Component(modules = {RetrofitModule.class})
public interface AppComponent {
    void inject(NewsListModelImp target);
}
Run Code Online (Sandbox Code Playgroud)

非常感谢任何建议,

R. *_*ski 5

Dagger-2重新工作.因此,如果注入了inside Activity(或Fragment)对象并且使用@Inject注释正确地对其构造函数进行了注释,那么构造函数的参数也将被注入.

假设您要注入的应用程序内部:

@Inject NyTimesPresenter presenter;

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    ((NYTimesSearchApplication) getApplication()).getAppComponent().inject(this);
}
Run Code Online (Sandbox Code Playgroud)

构造函数NyTimesPresenter必须注释为@Inject:

public class NyTimesPresenter {

    NewsListModelImp newsListModel;

    @Inject
    public NyTimesPresenter(NewsListModelImp newsListModel) {
        this.newsListModel = newsListModel;
    }
}
Run Code Online (Sandbox Code Playgroud)

NewsListModelImp构造函数还必须注释为@Inject:

public class NewsListModelImp implements NewsListModelContract {

    NYTimesSearchService mNYTimesSearchService;

    @Inject
    public NewsListModelImp(NYTimesSearchService nYTimesSearchService) {
        this.mNYTimesSearchService = nYTimesSearchService;
    }
}
Run Code Online (Sandbox Code Playgroud)

然后一切都会正确注入.

为什么参数应该作为构造函数的参数传递给类?这种设计模式符合SOLID principles.对象依赖项被注入到对象中而不是在其中创建,并且这样的代码很容易测试(在测试中依赖项可以被替换,即Mock.s)

额外信息:

可以注入实现特定接口的对象.这里描述这种技术.在你的情况下NyTimesPresenter可以有NewsListModelContract它的依赖而不是NewsListModelImp.为此,请为您添加另一个模块AppComponent:

@Singleton
@Component(
        modules = {
                RetrofitModule.class,
                AppModule.class
        })
public interface AppComponent {
Run Code Online (Sandbox Code Playgroud)

AppComponent 提供具体类实现接口的方法应如下所示:

@Singleton
@Module
public abstract class AppModule {

    @Binds
    public abstract NewsListModelContract provideNewsListModelContract(NewsListModelImp newsListModelImp);
}
Run Code Online (Sandbox Code Playgroud)

NyTimesPresenter应该改变的实现(只是用它实现的接口替换具体类):

public class NyTimesPresenter {

    NewsListModelContract newsListModel;

    @Inject
    public NyTimesPresenter(NewsListModelContract newsListModel) {
        this.newsListModel = newsListModel;
    }
}
Run Code Online (Sandbox Code Playgroud)