JSR-133によると、不変オブジェクトはスレッド セーフであり、同期は必要ありません。ただし、リフレクションを使用して final フィールドの値を更新することは可能です:
package com.stackoverflow;
import java.lang.reflect.Field;
public class WhatsGoingOn {
static class Immutable {
private final int value;
public Immutable(int value) {
this.value = value;
}
public int getValue() {
return value;
}
}
public static void main(String[] args) throws NoSuchFieldException, SecurityException, IllegalArgumentException, IllegalAccessException {
final Immutable immutable = new Immutable(Integer.MIN_VALUE);
final Field f = Immutable.class.getDeclaredField("value");
f.setAccessible(true);
System.out.println(immutable.getValue());
f.set(immutable, Integer.MAX_VALUE);
System.out.println(immutable.getValue());
}
}
リフレクションに依存するフレームワークの数 (Spring と Hibernate はほんの一部) を考えると、このシナリオについて仕様が何を示しているのか興味があります。たとえば、フィールドの更新を同期ブロックに入れると、他のスレッドでの可視性が保証されるか、値が final の仕様に従ってレジスタにキャッシュされます。
http://download.oracle.com/otndocs/jcp/memory_model-1.0-pfd-spec-oth-JSpec/