有一次面试被问到 String对象不可被改变么?
仔细一想是可变,然后研究了下发现了String的神奇之处!
不可变的String
众所周知String被设计为不可变
- 类被final修饰,不可被继承;
- 内部由一个char数组实现,由于该变量被final修饰所以保证了其长度不可变
- String没有提供任何修改char数组内容的方法,保证了内容不可被修改
改变String值
通过反射拿到char数组对象然后修改其值
String str = "123";
Field field = String.class.getDeclaredField("value");
field.setAccessible(true);
char[] value = (char[]) field.get(str);
value[0] = 'A';
System.out.println(str);
输出结果:
A23
发现String的神奇
jvm中实现了字符串常量池,改变一个值后相同的字符串会怎样?
String a = "123";
String b = "123";
Field field = String.class.getDeclaredField("value");
field.setAccessible(true);
char[] value = (char[]) field.get(a);
value[0] = 'A';
String c = "123";
System.out.println(a);
System.out.println(b);
System.out.println(c);
输出结果:
A23
A23
A23
String对象创建时都会先从字符串常量池中按字面值查找,如果有直接指向字符串常量,如果没有则创建对象及字符串常量
new 的方式传入字面值,会先处理字面值
替换常用的字面值
String str = "root";
Field field = String.class.getDeclaredField("value");
field.setAccessible(true);
char[] value = (char[]) field.get(str);
value[0] = 'f';
value[1] = 'u';
value[2] = 'c';
value[3] = 'k';
Map<Object, Object> map = new HashMap(4);
map.put("username", "root");
map.put("password", "123456");
System.out.println("username:" + map.get("username"));
System.out.println("password:" + map.get("password"));
输出结果:
username:fuck
password:123456