web-dev-qa-db-ja.com

Javaでは、finalフィールドをコンストラクタヘルパーから初期化できますか?

私は最後の非静的メンバーを持っています:

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...
}
42
csj

方法#2が最良のオプションです。問題は、プライベートメソッドに割り当てがある場合、それを呼び出すコンストラクターの外側のクラスの他のコードを妨げるものがないため、最終フィールドへの2番目の割り当てが試行されるという問題が発生することです。

Javaには、構築中にのみ呼び出すことができる個別のメソッドの構造はありません。

完全を期すために、3番目のオプションを作成できます。初期化時にマップを割り当ててから、ヘルパーメソッドにマップを設定させます。

 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...
  }

本当に混乱したい場合は、コンストラクタの代わりにイニシャライザを使用できますが、それを行うべきではないので、本当に知る必要がない限り、それについては説明しません。

15
Yishai

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.
}

必要に応じて、プライベートヘルパーコンストラクターの署名を変更できます(パラメーターデータを提供したり、署名をパブリックコンストラクターと区別したりするため)。

13
cjerdonek

オプション#2は、すべてのコンストラクター間で共有できるため、最も再利用可能なオプションです。ここで必要なのは、c#のコレクション初期化子です。 :)

(ところで:#3はコンパイルされません)

1
Simon