在Java中,可以从构造函数助手初始化final字段吗?

时间:2010-05-18 17:32:52

标签: java constructor initialization final

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

private final HashMap<String,String> myMap;

我想使用构造函数调用的方法初始化它。由于myMap是final,我的“helper”方法无法直接初始化它。我当然有选择:

我可以直接在构造函数中实现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很好,但是,我想知道是否有某种方法可以让helper方法直接操作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...
}

3 个答案:

答案 0 :(得分:13)

方法#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...
  }

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

答案 1 :(得分:13)

如何实现初始化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.
}

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

答案 2 :(得分:1)

选项#2是最可重用的选项,因为您可以在所有构造函数中共享它。我们在这里需要的是c#的集合初始化器。 :)

(顺便说一下:#3不会编译)