通常,最终的静态字段可能不会被修改。然而
System.in,System.out和
System.err是,由于遗留原因,必须允许通过方法来改变最终的静态字段
System.setIn,System.setOut和
System.setErr。我们称这些字段为写保护的,以区别于普通的最终字段。
编译器需要将这些字段与其他最终字段区别对待。例如,读取普通的最终字段对同步是“免疫”的:锁定或易失性读取中涉及的屏障不必影响从最终字段读取的值。由于可以看到写保护字段的值发生了变化,因此同步事件应该对其产生影响。因此,语义要求将这些字段视为不能由用户代码更改的普通字段,除非该用户代码在System类中。
顺便说一句,实际上,你可以
final通过调用反射
setAccessible(true)(或使用
Unsafe方法)来通过反射使它们发生变化。Hibernate和其他框架等在反序列化过程中使用了此类技术,但它们有一个局限性:修改前已看到final字段值的代码不能保证在修改后会看到新值。有问题的字段的特殊之处在于它们不受此限制,因为它们由编译器以特殊方式处理。



