问题来源
在公司写业务代码时,需要写一个函数对字符串做操作,总所周知,java中字符串是引用数据类型,引用数据类型在传入函数时,对象只需在函数中做操作而不必返回该对象也可以实现修改,这个是不是和C语言中的指针一样,其实java中我们声明的对象变量也相当于一个指向堆中数据的指针
可就是这样,当我在函数中对字符串做了修改,但是实际该字符串并没有变化
看看代码
代码
代码功能是定义一个字符串,然后经过函数处理,进行输出
public class StringDemo {
public static void main(String[] args) {
String hello = "hello";
System.out.println("函数处理前hello: "+hello);
convert(hello);
System.out.println("函数处理后hello: "+hello);
}
private static void convert(String str) {
System.out.println("函数处理前str: "+str);
str += " world";
System.out.println("函数处理后str: "+str);
}
}
实际的输出
解决问题
问题
既然函数中的str和hello变量指向的对象是一样的,那么为什么会str修改成功,而hello没有变化
错因排查
如果str和hello指向相同,hello就应该是hello world,那么出问题的原因可能是str和hello指向变了
原理解析
回顾java中字符串对象的创建,我们知道只要不是new的字符串,对于
String a = "str";
String b = "str";
a和b他们的指向是一样的,即a==b的输出为true,因为a和b都是会去唯一字符串常量池中取对象,而一旦有一个是new出来的对象,无论值是否相当,a==b的输出永远是false
当然使用intern方法可以把new的字符串放进字符串常量池中
所以我们也可以知道函数中变量指向的变化了,如下图所示
- 第一步str和hello一样指向字符串常量池中hello这个变量
- 第二步str+=" world"可以等价于str = str + " world",这里str的指向就变化了,首先会在常量中新建" world"这个字符串,然后拼接这两个字符串得到一个新的字符串,也就是说str现在是指向这个新的字符串的,所以hello对象的指向从头到尾都没有变化,变的只有hello
总结
字符串的大多数问题都是由唯一字符串常量池这个机制引出的,所以理解号字符串常量池非常重要
虽然这个问题很简单,背后涉及的机制也很浅显,在工作也是很平常的工作,可就是太平常了,如果不懂的话在工作中就会出错,公司项目一般较大,排查就需要很多时间,所以勿以问题简单而不注意