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)
非常感谢任何建议,
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)
| 归档时间: |
|
| 查看次数: |
1038 次 |
| 最近记录: |