有几种不同的方法可以初始化复杂对象(使用注入的依赖项和所需的注入成员设置),看起来都很合理,但各有优缺点。我举一个具体的例子:
final class MyClass {
private final Dependency dependency;
@Inject public MyClass(Dependency dependency) {
this.dependency = dependency;
dependency.addHandler(new Handler() {
@Override void handle(int foo) { MyClass.this.doSomething(foo); }
});
doSomething(0);
}
private void doSomething(int foo) { dependency.doSomethingElse(foo+1); }
}
如您所见,构造函数做了 3 件事,包括调用实例方法。我被告知从构造函数调用实例方法是不安全的,因为它绕过了编译器对未初始化成员的检查。即我可以doSomething(0)
在设置之前调用this.dependency
,这会编译但不起作用。重构它的最佳方法是什么?
制作
doSomething
静态并显式传递依赖项?在我的实际情况中,我有三个实例方法和三个成员字段,它们都相互依赖,所以这似乎是使所有这三个静态化的大量额外样板。将
addHandler
and移动doSomething
到一个@Inject public void init()
方法中。虽然与 Guice 一起使用是透明的,但它需要任何手动构造以确保调用init()
,否则如果有人忘记,该对象将无法完全发挥作用。此外,这暴露了更多的 API,这两者似乎都是坏主意。包装一个嵌套类以保持依赖关系,以确保它在不暴露额外 API 的情况下正常运行:
类依赖管理器{ 私有最终依赖依赖; 公共 DependecyManager(依赖依赖) { ... } 公共 doSomething(int foo) { ... } } @Inject public MyClass(依赖依赖){ DependencyManager manager = new DependencyManager(dependency); manager.doSomething(0); }
这会将实例方法从所有构造函数中提取出来,但会生成一个额外的类层,当我已经有内部和匿名类(例如那个处理程序)时,它可能会变得混乱 - 当我尝试这个时,我被告知将其移动DependencyManager
到一个单独的文件,这也令人反感,因为它现在是多个文件来做一件事情。
那么处理这种情况的首选方法是什么?