今天咱们一起来说说java家族中的使用频率最广泛的String
-
String的特性
在Sring源码的类注释当中有如下一段话:
这说明了String的一个重要特性,String 是value不可改变的
然而String类还使用了final来修饰,表明String的第二个重要特性,** String是不可被继承的** String的定义方法
一般我们在代码中可以使用如下三种方法定义String对象
String str1 = new String("cat");
String str2 = "cat";
String str3 = "c"+"at";
String str4 = str1+"cat";
第一种方式直接通过关键字new创建对象,毫无疑问str1指向堆内存
第二种直接赋值,Str2指向常量池
第三种通过+号连接赋,最终str3也是指向常量池
第四种是带三种的延伸,这个最终str4是指向堆内存
对于前三种方式大家应该都没什么疑问,那么第四种方法为什么也会指向堆内存呢?欲知真相还需从字节码入手啊!
先来看看几个经典的面试题
- First
String s1 = "abc";
String s2 = "abc";
System.out.println("s1 == s2 : " + (s1 == s2)); //true
So esay,对吧!
- Second
String s1 = "abc";
String s2 = new String("abc");
String s3 = new String("abc");
System.out.println("s2 == s3 : " + (s2 == s3)); //false
System.out.println("s1 == s3 : " + (s1 == s3));//false
还是很简单,对么,只要有点基础的人都知道答案呢!
- Third
public class Test{
public static void main(String[] args){
String str1 = "ab" + "cd"; //
String str11 = "abcd";
//System.out.println("str1 == str11 : " + (str1 == str11)); //true
String str2 = "ab"+2;
String str22 = "ab2";
//System.out.println("str2 == str22 : " + (str2 == str22)); //true
final String str3 = "ab";
String str31 = str3+"cd";
String str32 = "abcd";
//System.out.println("str2 = str3 : " + (str2 == str3)); // false
}
}
这个对新手来说可能就有点不知所措了,来来来,上我们的终极利器,请看下面编译之后的字节码
因为常量的值在编译期间就已经确定了,这里"ab","cd","2"都是常量,所以编译器会自动对
String str1 = "ab" + "cd";String str2 = "ab"+2;
进行优化,效果相当于String str1 = "abcd";String str2 = "ab2";
这下你明白为什么结果会是true了吧
- Fourth
public class Test{
public static void main(String[] args){
String str2 = "ab";
String str3 = "cd";
String str41 = str2 + str3;
String str42 = "ab"+str3;
String str43 = str2+"cd";
String str5 = "abcd";
//System.out.println("str41 = str5 : " + (str41 == str5)); // false
//System.out.println("str42 = str5 : " + (str42 == str5)); // false
//System.out.println("str43 = str5 : " + (str43 == str5)); // false
}
}
这个跟例三差不多,只不过例三是常量,这里变成了变量,结果就迥然不同,下面我们来看看这段代码的字节码长啥样子
发现虚拟机在处理变量字符串用+号连接的时候是生成了一个StringBuilder对象,然后调用StringBuilder的append方法,最后在调用StringBuilder的toString方法返回。StringBuilder的toString方法源码如下:
这下就彻底明白了吧。它是从新new一个String对象,结果当然为false了!
针对==
的总结如下
众所周知,==
是判断内存地址的,那么我们判断的话看引用最终指向哪里不就可以了么
- new出来的对象是存放在堆内存中的,引用指向堆内存,那么地址肯定是唯一的了,所以结果肯定为false
- 直接声明的字符串引用是指向常量池的,而常量池中的相同内容的字符串只有一份,所以结果为true
- 通过+号连接的字符串分如下两种情况
3.1 如果+号两边连接的是常量,那么会在编译期间进行优化,结果同2
3.2 如果+号两边连接的有变量,不管是new出来的也好,直接声明的也好,java虚拟机执行的时候都会生成一个StringBuilder对象sb,然后调用sb.apend()
方法,最后通过sb.toString()
返回一个新的字符串。那么此时引用就指向堆内存,结果同1
String.intern()
当一个String实例str调用intern()方法时,虚拟机会查找常量池中是否有相同Unicode的字符串常量,如果有,则返回其的引用;如果没有,则在常量池中增加一个Unicode等于str的字符串并返回它的引用;PS:我并没有想到什么时候会没有
public class Test{
public static void main(String[] args){
String s00 = "kvill";
String s11 = new String("kvill");
String s22 = new String("kvill");
System.out.println( s00 == s11 ); //false
s11.intern(); //虽然执行了s1.intern(),但它的返回值没有赋给s1
s22 = s22.intern(); //把常量池中"kvill"的引用赋给s2
System.out.println( s00 == s11); //flase
System.out.println( s00 == s11.intern() ); //true 说明s11.intern()返回的是常量池中"kvill"的引用
System.out.println( s00 == s22 ); //true
}
}
关于String,StringBuffer,StringBuilder的区别
- String是值不可变的,每次进行连接操作是都是返回一个新的String对象,StringBuffer,StringBuilder是值可变的,操作是返回的是this
这也就是为什么在进行大量字符串连接运算时,不推荐使用String,而推荐StringBuffer和StringBuilder。 - StringBuffer是线程同步的,安全性高,但执行效率低
StringBuilder是非线程同步的,安全性低,但执行效率高
这是StringBuffer,StringBuilder append方法的源码
THE END
如有错误之处还请大家不吝赐教,多多指正,共同进步!