01-改变字符串s的内容
一、题目
看以下代码回答问题:
public static void main(String[] args) {
String s = new String("abc");
// 在中间可以添加n行代码,必须保证s的引用的指向不变,最终输出变成abcd
System.out.println(s);
}
二、解决
首先看到题目要求,给人的第一印象感觉很简单,其实就是将字符d拼接到原来的字符串s上就可以完成要求了,但是具体还得看我们如何实现。
①append()
第一种想法,首先我们会想到StringBuilder的append()方法将字符d拼接上去,其实这种操作是不可取的,看以下代码
public static void main(String[] args) {
String s = new String("abc");
// 在中间可以添加n行代码,必须保证s的引用的指向不变,最终输出变成abcd
StringBuilder builder = new StringBuilder(s);
// StringBuilder d = builder.append("d");
builder.append("d");
System.out.println(s);
}
经过控制台输出测试后,这种方法是不可取的,因为append()方法执行后其实还是一个新的StringBuilder对象,原来的String s对象并没有改变,所以这种方法不可取。
②replace()
然后还有很多小伙伴会想到String类的replace()方法,如下代码。
public static void main(String[] args) {
String s = new String("abc");
// 在中间可以添加n行代码,必须保证s的引用的指向不变,最终输出变成abcd
// String replace = s.replace("abc", "abcd");
s.replace("abc","abcd");
System.out.println(s);
}
事实上,这种方法也是不行的,最简单的方式可以直接看到,replace后直接生成了一个新的String对象,而原来的String s对象是没有改变的。
其实这就涉及到replace方法的底层原理了,replace()方法的底层原理是通过new StringBuffer()进行替换的原理的,这种方法也不能改变对象s的内容。
③反射
事实上,了解过String的小伙伴们都知道,String是不可变的,底层是一个由final修饰的char[]数组进行实现的,所以我们没有办法改变String对象的内容,但是我们可以通过反射的方式获取到String对象中的char[]数组,并修改其属性,进而改变对象的值。
代码如下:
public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException {
String s = new String("abc");
// 在中间可以添加n行代码,必须保证s的引用的指向不变,最终输出变成abcd
Field value = s.getClass().getDeclaredField("value");
value.setAccessible(true);
value.set(s, "abcd".toCharArray());
System.out.println(s);
}