【发布时间】:2010-03-24 21:50:04
【问题描述】:
有几种不同的方法可以初始化复杂对象(使用注入的依赖项和所需的注入成员设置),看起来都很合理,但各有优缺点。我举个具体的例子:
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 件事,包括调用实例方法。我被告知从构造函数调用实例方法是不安全的,因为它绕过了编译器对未初始化成员的检查。 IE。我可以在设置this.dependency 之前调用doSomething(0),这样可以编译但不起作用。重构它的最佳方法是什么?
将
doSomething设为静态并显式传入依赖项?在我的实际案例中,我有三个实例方法和三个成员字段,它们都相互依赖,所以这似乎需要大量额外的样板来使这三个都成为静态。将
addHandler和doSomething移动到@Inject public void init()方法中。虽然与 Guice 一起使用是透明的,但它需要任何手动构造以确保调用init(),否则如果有人忘记,该对象将无法完全发挥作用。此外,这会暴露更多的 API,这两者似乎都是坏主意。-
包装一个嵌套类以保持依赖关系,以确保它正常运行而不会暴露额外的 API:
类 DependencyManager { 私有最终依赖依赖; 公共 DependecyManager(依赖依赖) { ... } 公共 doSomething(int foo) { ... } } @Inject public MyClass(依赖依赖){ DependencyManager manager = new DependencyManager(dependency); manager.doSomething(0); }这会将实例方法从所有构造函数中提取出来,但会生成一个额外的类层,当我已经有内部和匿名类(例如那个处理程序)时,它可能会变得混乱 - 当我尝试这个时,我被告知将DependencyManager移动到一个单独的文件,这也令人讨厌,因为它现在是多个文件来做一件事情。
那么处理这种情况的首选方法是什么?
【问题讨论】:
-
@Steve:我刚刚删除了第一个“pre”标签,以便代码使用颜色编码语法显示:)
-
酷,不知道它是这样工作的。
标签: java constructor guice