2

我最近一直在玩弄下面的 java 模式,它帮助解决了一些代码组织问题,但我有多个关于最佳实践、性能等的问题。

public class Example {

    private static int[] somethingList = int[128];

    public static final Example somethingOne = new Example(1);
    public static final Example somethingTwo = new Example(2);
    public static final ExampleChild somethingThree = new ExampleChild(3); // ExampleChild would extend Example

    private Example( int id ){
        // checks for existing id, etc
        somethingList[id] = this;
    }

    public static Example getById( int id ){
        return somethingList[id];
    }
}

该类将事物列表注册到静态变量,但有一个私有构造函数也将通过 id # 的列表保存到数组中。Minecraft 是在野外使用它的一个例子 - 用于他们的项目类“注册表”。

几个问题:

  • 如果在此之外的任何地方都没有保存 Example 的单个实例,是否调用Example.somethingOneExample.getById()重新运行所有这些代码,或者这一切只运行一次?也许我缺少一些理解static
  • 您是否有任何理由希望somethingOne公开的静态实例,而不是仅仅通过 id 调用它?getById如果您有 mods/plugins 将自定义项目添加到此处没有按名称静态引用的列表中,这似乎是友好的。
4

2 回答 2

4

如果在此之外的任何地方都没有保存 Example 的单个实例,请调用 Example.somethingOne 或 Example.getById() 重新运行所有这些代码,还是只运行一次?也许我缺少对静态的一些理解?

不,静态块在类加载时运行,并与代表类本身的 java.lang.Class 的实例相关联。

类是延迟加载的,但一旦加载,就不会在类加载器的生命周期内重新加载。

调用静态方法或访问静态变量而不是使用它的实例不会降低性能。

有什么理由让你希望 somethingOne 的静态实例是公开的,而不是仅仅通过 id 调用它?如果您有 mods/plugins 将自定义项目添加到此处没有按名称静态引用的列表,似乎 getById 是友好的。

你在这里回答了你自己的问题。通过方法运行访问可以在将来添加模块/插件或其他一些工作,而无需重构调用代码。

然而,我可能希望实例成为静态成员的一个原因是,如果调用代码需要与特定实例而不是其他实例交互,并且您只是希望您的类能够控制这些实例是什么,也许这样您就可以添加新的需要时的实例。

枚举就是一个很好的例子。

我已经将您上面的确切模式用于动态枚举,我想要的是一个枚举,但我需要某些实例是子类型并且客户不必担心它。

于 2013-09-08T04:47:32.983 回答
0

由于调用私有构造函数的唯一位置是 and 的静态声明somethingOnesomethingTwo因此该构造函数将在第一次使用类时调用,这将触发类的加载和静态代码的执行。

这将在每次初始化类时执行,这意味着在您使用 Java 提供的默认类加载的典型场景中,无论您访问字段多少次,它都只会被调用一次。

您可能需要阅读JLS 12.4 Initialization of Classes and InterfacesJVMS 5.5 Initialization

不确定为什么静态实例是公共的,它们完全可以是私有的 static final 并且通过在第一次调用 getId 方法时触发类的初始化并获得相同的结果。也许这只是方便,毕竟所有字段都是静态和最终的,那么为什么不提供对它们的直接访问呢?

于 2013-09-08T04:44:06.927 回答