我们都知道String是java中一个不可变类,因为String内部是一个final修饰的char数组:
private final char value[];
由于value是私有的final变量,String中也没有提供get和set方法,使得value无法改变。
但是value是一个引用,就像c++中的指针一样,指向一个数组内存的地址,被final修饰无法改变他的引用指向,但是我们可以直接改变内存数组中的数据,value是私有的,获取就需要用到反射:
public static void change(String s) throws Exception{
System.out.println("before: "+s);
Field valueField = s.getClass().getDeclaredField("value");
valueField.setAccessible(true);
char[] temp = (char[])valueField.get(s);
temp[0] = 'L';
System.out.println("after: "+s);
}
结果:
//输入abcd
before: abcd
after: Lbcd
我们使用Field的setAccessible方法使得private的变量变的可以访问,获取到数组,直接更改数组中的值,但这样破坏了类的访问规则,会产生很多安全隐患,知乎上看到一个回答:
setAccessible是一种hack
潜台词是:你清楚内部实现,你知道你在做什么,相信你不会搞砸,平时不需要过分担心“如果别人用setAccessible来搞我怎么办”之类的问题,死代码防不了活人,也没必要
当然,java将String设置为不可变是有道理的,暴力的更改会带来很多问题,例如:
String s1 = "abcd";
String s2 = "abcd";
Field valueField = s1.getClass().getDeclaredField("value");
valueField.setAccessible(true);
char[] temp = (char[])valueField.get(s1);
temp[0] = 'L';
System.out.println(s1);
System.out.println(s2);
只改变s1的值,s2也变了,由于字符串存储在方法区常量池中,s1和s2指向同一块内存区域,通过s1改变了字符串常量的值,s2也随着改变了。
所以不到万不得已,不要随便通过暴力手段更改不可变类的值,会有很多难以发现的问题存在。