我正在关注我的Android应用程序的MVC模式,并且已经遇到过这个问题几次并且不得不解决它。 当我的应用程序能够使用字段上的@Inject注释创建注入对象时,该对象@Inject字段为空,通常会导致崩溃。例如,我有控制器类将处理逻辑和流动。任何片段/活动都将回调到其控制器以通知用户交互/状态更改。但是,注入的Controller实例通常为空。
我将举一个简单的例子来说明。下面,Controller正在创建一个注入活动,然后使用它通过添加片段来启动流程。处理该依赖关系但是对控制器的活动依赖性不是(即为空)。
用于处理业务逻辑和流程的Simple Controller类:
public class SomeController {
@Inject
SomeActivity someActivity;
private SomeComponent component;
private final Application app;
@Inject
public SomeController(Application app) {
this.app = app;
}
private void startActivity() {
component = Dagger_SomeComponent().builder()
.someModule(app)
.build();
someActivity.getFragmentManager().beginTransaction().
.add(R.id.content, SomeFragment.class, null)
.commit();
}
public void activityStarted() {
//callback when Activity is ready...
}
}
处理用户交互并回调控制器以执行某些业务逻辑的简单活动:
public class SomeActivity extends Activity {
@Inject
SomeController controller;
private void controllerCallback() {
//notify controller of something here...
}
}
用于将对象注入图形的简单模块类:
@Module
public class SomeModule {
private Application app;
public SomeModule(Application app) {
this.app = app;
}
@Provides
@Singleton
SomeController provideSomeController( return new SomeController(app); )
@Provides
SomeActivity provideSomeActivity( return new SomeActivity();)
}
Simple Component类,用于提供使用对象的方法:
@Component
public interface SomeComponent {
void addController(SomeController controller);
SomeController controller();
SomeActivity activity();
}
答案 0 :(得分:1)
我相信你的例子中遗漏了一些东西:
1)Dagger2 Component是一些提供依赖关系的模块和一些“消耗”依赖关系的注入点之间的桥梁。您的组件应至少有一个链接模块,注释应如下所示:
@Component(modules = SomeModule.class)
2)使用'new'运算符的模块中的SomeActivity实例化没有任何意义。好的,您可以创建Activity对象,但谁将管理它的状态,调用它的生命周期方法等?如果你真的想要将现有Activity的引用作为依赖项传递 - 它是可能的,但是以不同的方式。例如,创建应用程序级模块&组件,以及单独的活动级别模块&组件,将现有的Activity引用作为模块构造函数参数传递。
3)Android框架创建活动,因此您应该在活动中使用字段注入。向您的组件添加如下行:
void inject(SomeActivity activity);
将创建组件的代码放入Activity。例如onCreate():
SomeComponent component = Dagger_SomeComponent().builder()
.someModule(getApplication())
.build()
.inject(this);
4)请尽量避免@Singleton注释。首先要确保您声明的所有依赖项都已满足,并且您已经不再为null。接下来,您可以检查一些现有的Dagger2开源项目,以进行正确的范围注释。
答案 1 :(得分:1)
我遇到了这个问题,并使用this accepted answer解决了这个问题。
TLDR:我必须在Component中添加
inject()
方法才能完全正确 类型,而不仅仅是inject(SomeSuperclass foo)
基本上,我有一个需要注射的活动:
class SubActivity extends BaseActivity {
@Inject Type someVar;
@Override
protected void onCreate(Bundle bundle) {
getComponent().inject(this);
}
}
但我的组件看起来像:
@Component
public interface MyComponent {
void inject(BaseActivity activity); // This allows SubActivity to compile but messes up injection
}
解决方案:为确切的子类型添加注入方法:
@Component
public interface MyComponent {
void inject(BaseActivity activity);
void inject(SubActivity activity); //<-- This solved it
}
答案 2 :(得分:0)
在我的情况下,depends-on =“ bean1”在应用程序上下文中的property-placeholder内引起了问题。我使用@PostConstruct进行注入工作。