1.string是final类,无法继承,string的值无法改变。
string a = “abc”;
a = a+“de”;
这一过程中,先在堆中开辟内存存储“abc”,引用变量a在栈中保存abc的地址;开辟内存保存“de”,开辟内存保存“abcde”,然后将a的地址改成abcde的地址。
所以在交换方法中,虽然你传递了string变量的地址,但是你无法真正改变string所保存的地址。虽然在swap方法中,你使用了a = a+“de”;,但是,方法之外,a变量还是指向abc。方法内的a指向的是abcde,保存abcde的地址。
1.2使用new创建string
new创建的和直接赋值的,使用==比较肯定是false。因为new必须在堆中创建字符,如果字符串常量池没有,还要在常量池中创建。
String s1 = "Programming";
String s2 = new String("Programming");
String s3 = "Program" + "ming";
System.out.println(s1 == s2);//false,
//一个可能从常量池中取对象,另一个直接在堆中创建
System.out.println(s1 == s3);//true
s3符合第一条的过程:先得到这个"Program",在创建 "ming",
最后开辟内存创建"Programming",但是这个已经有了,所以不用创建,直接引用地址。
String a1 = "ABC";
String b1 = "AB";
String c2 = b1 + "C";
System.out.println( a1 == c2 );//false
但是,此处涉及变量后,就是false,说明他们的地址不同了。
有人说,string的+是这么操作的:
String b = "a" + bb;
String b = new StringBuilder().append("a").append(bb).toString();
这是tostring的源码:
public String toString() {
// Create a copy, don't share the array
return new String(value, 0, count);
}
确实创建了一个new string,但是,这样一来,
System.out.println(s1 == s3);//true
就无法解释了啊!!!!!@!!
最终答案:“String+变量”因为编译时无法进行优化,所以这一条语句的操作是在运行时进行的,且会产生新的对象,而不是直接从jvm的string池中获取。
也就是说:+的两边是字符串,那么在编译时确定;+的两边有变量,在运行时确定,使用new StringBuilder添加,最后转换为new string。
2.所以推荐使用stringBuild和stringBuffer
StringBuffer 字符串变量(线程安全)
StringBuilder 字符串变量(非线程安全)
3.想要交换,除了传递可以改变对象的引用(基本类型和string都不行),还可以传递数组。