java - 从 Java 构造函数调用实例方法是好是坏?

标签 java constructor guice

有几种不同的方法可以初始化复杂对象(注入(inject)依赖项和注入(inject)成员的必需设置),这些方法看起来都很合理,但各有优缺点。我举一个具体的例子:

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),这会编译但不起作用。重构它的最佳方法是什么?

  1. doSomething 设为静态并显式传递依赖关系?在我的实际案例中,我有三个实例方法和三个成员字段,它们都相互依赖,所以这似乎是很多额外的样板来使所有这三个都是静态的。

  2. addHandlerdoSomething 移动到 @Inject public void init() 方法中。虽然与 Guice 一起使用将是透明的,但它需要任何手动构造以确保调用 init() 否则如果有人忘记该对象将无法完全发挥作用。此外,这会暴露更多的 API,这两个想法似乎都是坏主意。

  3. 包装一个嵌套类以保持依赖性,以确保它在不暴露额外 API 的情况下正常运行:

    class DependencyManager {
      private final Dependency dependency;
      public DependecyManager(Dependency dependency) { ... }
      public doSomething(int foo) { ... }
    }
    @Inject public MyClass(Dependency dependency) {
      DependencyManager manager = new DependencyManager(dependency);
      manager.doSomething(0);
    }
    这会从所有构造函数中提取实例方法,但会生成一个额外的类层,当我已经有了内部类和匿名类(例如那个处理程序)时,它可能会变得困惑——当我尝试这个时,我被告知要移动 DependencyManager 到一个单独的文件,这也是令人反感的,因为它现在是多个文件来做一件事情。

那么处理这种情况的首选方法是什么?

最佳答案

Effective Java 中的 Josh Bloch 建议使用静态工厂方法,尽管我找不到针对这种情况的任何论据。然而,Java Concurrency in Practice 中也有类似的情况。 ,专门用于防止从构造函数中泄漏对 this 的引用。应用于这种情况,它看起来像:

final class MyClass {
  private final Dependency dependency;

  private MyClass(Dependency dependency) {
    this.dependency = dependency;
  }

  public static createInstance(Dependency dependency) {
    MyClass instance = new MyClass(dependency);
    dependency.addHandler(new Handler() {
      @Override void handle(int foo) { instance.doSomething(foo); }
    });
    instance.doSomething(0);
    return instance;
  }
  ...
}

但是,这可能不适用于您使用的 DI 注释。

关于java - 从 Java 构造函数调用实例方法是好是坏?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2511665/

相关文章:

c++ - 奇怪的对象分配行为 c++

java - 在java中加入多个迭代器

java - Java中的并行赋值?

c++ - 为什么我的默认构造函数返回一个 boolean 值?

java - 将 ENUM 类型分配给构造函数

java - Guice @Provides提供依赖于两个参数的类

java - 运行 Nashorn JavaScript 的 Gradle 任务

java - JPA 中的 transient 字段和查询中的设置

java - 启动嵌入式 Jetty 不起作用,SLF4J 响应错误

java - HK2 相当于 Guice 中的 @Provides for Jersey 2