在 Java 中,可以从构造函数帮助程序初始化最终字段吗?

2022-09-01 00:39:32

我有一个最终的非静态成员:

private final HashMap<String,String> myMap;

我想使用构造函数调用的方法初始化它。由于 myMap 是最终的,因此我的“帮助程序”方法无法直接初始化它。当然,我有选择:

我可以直接在构造函数中实现 myMap 初始化代码。

MyConstructor (String someThingNecessary)
{
    myMap = new HashMap<String,String>();

    myMap.put("blah","blahblah");
    // etc...

    // other initialization stuff unrelated to myMap
}

我可以让我的帮助器方法生成 HashMap,将其返回给构造函数,然后让构造函数将对象分配给 myMap。

MyConstructor (String someThingNecessary)
{
    myMap = InitializeMyMap(someThingNecessary);

    // other initialization stuff unrelated to myMap
}

private HashMap<String,String> InitializeMyMap(String someThingNecessary)
{
    HashMap<String,String> initializedMap = new HashMap<String,String>();

    initializedMap.put("blah","blahblah");
    // etc...

    return initializedMap;
}

方法#2很好,但是,我想知道是否有某种方法可以允许帮助器方法直接操作myMap。也许是一个指示它只能由构造函数调用的修饰符?

MyConstructor (String someThingNecessary)
{
    InitializeMyMap(someThingNecessary);

    // other initialization stuff unrelated to myMap
}


// helper doesn't work since it can't modify a final member
private void InitializeMyMap(String someThingNecessary)
{
    myMap = new HashMap<String,String>();

    myMap.put("blah","blahblah");
    // etc...
}

答案 1

方法#2是您的最佳选择。问题在于,如果在私有方法中有一个赋值,则没有什么可以阻止调用它的构造函数之外的类中的其他代码,这将导致尝试对最终字段进行第二次赋值的问题。

Java没有只能在构造期间调用的单独方法的构造。

为了完整起见,我们可以创建第三个选项,在初始化时分配映射,然后让帮助器方法填充它:

 private final HashMap<String, String> myMap = new HashMap<String, String();

然后:

 MyConstructor (String someThingNecessary)
 {
    initializeMyMap(someThingNecessary);

    // other initialization stuff unrelated to myMap
 }


 // helper doesn't work since it can't modify a final member
 private void initializeMyMap(String someThingNecessary)
 {

     myMap.clear();
    myMap.put("blah","blahblah");
    // etc...
  }

如果你真的想混淆,你可以使用初始值设定项而不是构造函数,但你不应该这样做,所以除非你真的需要知道,否则我不会扩展它。


答案 2

如何实现一个初始化HashMap的私有构造函数,然后让你的主构造函数调用该私有构造函数?

例如——

// Helper function to initialize final HashMap.
private MyConstructor()
{
    myMap = new HashMap<String,String>();
    myMap.put("blah","blah");
}

MyConstructor (String someThingNecessary)
{
    // Initialize the HashMap.
    this();
    // Other initialization code can follow.
}

您可以根据需要修改私有帮助器构造函数的签名(例如,提供参数数据或使签名与任何公共构造函数不同)。