String类:
其在java.lang包下,因而使用不需要导包。
String 类都是字符串类,所以说Java里面所有的双引号字符串都是String类对象。
字符串不可变,他们的值在创建会就不可改变了。
String常见构造方法:
一个问题,看以下代码,判断两个打印的结果
第一个是TRUE,第二个是FALSE,因为s1g和s2都是对象池里面的,所以判断会相等。
s3和s4是数组类型强转过来的,他们的值虽然相等,但是比较的东西却是地址,所以打印的结果会是FALSE
创建字符串对象的区别对比:
以" ”这种形式创建的字符串数据,只有字符串相同(大小写和顺序都完全相同),无论在代码中出现几次,都只会建立一个String对象,并在常量池中维护。
字符串常量池:当使用双引号创建字符串对象时,系统会检查该字符串是否在字符串常量池中存在,如果存在,就会复用,而不会重新创建,而不存在的时候就会直接创建。
另外,每次通过new创建的字符串对象,每一次new都会申请一个内存空间,即便内容相同,但是地址不同。
一个很经典的题目:
可能很多人都觉得两个打印的结果都是TRUE,但是事实并非如此,第一个为FALSE,第二个为TRUE,上面的代码中,s1等于abcd,此时字符串常量池里面就存在“abcd”的字符串,再来看s6="ab"+"c"+"d",因为Java里面存在常量优化机制,所以会计算右边的结果然后赋值给左边,实际上就等于"abcd",而"abcd"在字符串常量池里面已经有了,所以会把他的引用赋值给s1,所以两个的地址值是相等的。
但是s5=s2+"c"+"d",因为等于号右边存在变量s2所以在有字符串变量和"+"的时候实际上在底层是调用的StringBuilder的append()方法进行拼接,然后调用toString();方法再变成String字符串,放在堆内存中,并不在字符串常量池,所以s1==s5的结果是false。
再看一段代码:
public static void main(String[] args) {
String s1 = "abc1";
String s2 = "abc" + 1;
String s3 = "abc1.6";
String s4 = "abc" + 1.6;
String s5 = "abctrue";
String s6 = "abc" + true;
System.out.println(s1 == s2);
System.out.println(s3 == s4);
System.out.println(s5 == s6);
}
这里的三个结果都是TRUE,因为只有字符串在前面,后面用加号拼接的默认都是加了双引号的字符串常量,然后根据常量优化机制将右边计算出结果,不存在就添加,存在就将常量池里面的引用赋值给变量。
再看一段代码:
结果是FALSE,因为s10的数字在前面,所以数字会先相加,因而s10的结果是6abcd,所以判断自然是FALSE。