String是不可变的*,但这仅意味着你无法使用其公共API对其进行更改。
你在这里所做的是使用反射来绕过常规API。同样,你可以更改枚举的值,更改整数自动装箱中使用的查找表等。
现在,原因s1和s2变化值是它们都引用相同的实习字符串。编译器执行此 *** 作(如其他答案所述)。
原因s3实际上并不令我惊讶,因为我认为它可以共享
value数组(它在Java的较早版本中(在Java 7u6之前)已完成)。但是,查看的源代码
String,我们可以看到
value实际上已复制了子字符串的字符数组(使用
Arrays.copyOfRange(..))。这就是为什么它保持不变。
你可以安装SecurityManager,以避免恶意代码执行此类 *** 作。但是请记住,某些库依赖于使用这些反射技巧(通常是ORM工具,AOP库等)。
*)我最初写道
Strings并不是真正不变的,只是“有效的不变”。这可能会在的当前实现中产生误导String,其中
value确实标记了数组
private final。但是,仍然值得注意的是,没有办法在Java中将数组声明为不可变的,因此即使使用适当的访问修饰符,也必须注意不要将其暴露在类之外。
由于这个话题似乎非常受欢迎,因此建议你进一步阅读以下内容:Heinz Kabutz在JavaZone 2009上发表的《 Reflection Madness》演讲,其中涵盖了OP中的许多问题以及其他反思……嗯……疯狂。
它涵盖了为什么有时有用。为什么,在大多数情况下,你应该避免使用它。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)