jiava 字符串分为三种类型:String,StringBuffer, StringBuilder。接下来我们来看看这三者的区别。
String 初始时默认为空字符序列。
方式一:java
代码解读复制代码String str1 = new String("abc");
方式二:
代码解读复制代码String str2 = "abc";
方式一先在堆内存中创建字符串常量 “abc”,然后再新创建 String 对象,对象里的内容为字符串常量 “abc” 的地址。str1 为新建对象的地址。
方式二:str2 直接指向堆内存中的字符串常量 “abc”。
String 为字符串常量,是因为 String 对象一旦创建后,就不能更改。java
代码解读复制代码String str = new String("abc");
str = str + "de";
System.out.println(str); // 输出 abcde
以上代码看似 str 的值能够改变,但由底层原理来看,其实不然。
首先先创建一个 String 对象 str,然后把 “abc” 的地址赋给 str。第二行时,在字符串常量池中添加 “abc” + "de" 字符串,并再创建一个新的 String 对象,新对象的内容为字符串常量池中 “abcde” 的地址值,然后 str = 此 String 对象,原 String 对象被垃圾回收机制回收掉。所以,Java 中对 String 对象进行的操作实际上是一个不断创建新的对象并且将旧的对象回收的一个过程。
由于 String 是不可变类,所以在多线程中使用是安全的。且其不可变,所以在 java 运行时节省大量堆空间。
因为 String 不可变,当创建字符串时,它的 hashcode 被缓存下来,不需要再次计算。又因为 HashMap 内部实现是通过 key 的 hashcode 来确定 value 的存储位置,多以相比于其他对象更快。
StringBuilder 和 StringBuffer 初始时默认容量为 16 个字符,
StringBuilder 和 StringBuffer对象只有一种创建方式:java
代码解读复制代码StringBuffer str1 = new StringBuffr("abc");
StringBulider str2 = new StringBuilder("abc");
StringBuilder 和 StringBuffer 都为字符串变量,里面的内容可以随意更改且其指向的地步不会发生改变。java
代码解读复制代码StringBuffer str = new StringBuffer(); // 创建一个空的 StringBuffer 对象。
str.append(“abcde”); // 添加字符串
str.delete(3, 5); // 删除 de
StringBuilder 的方法中没有使用 synchronized 关键字修饰,所以为非线程安全。但其效率较高。
StringBuffer 的很多方法用 synchronized 修饰,意味着多个线程只能互斥地调用这个方法。
适用环境:
String:适用于少量的字符串操作。
StringBuilder:适用于单线程下在字符缓冲区进行大量的操作。
StringBuffer:适用于多线程下在字符缓冲区进行大量的操作。
执行速度:
StringBuilder 快于StringBuffer 快于 String
由于每次改变字符串时,String 都要新建一个对象,在赋值,而其他两个直接赋值,故 String 最慢。
由于 StringBuilder 是非线程安全的,当使用在多线程的环境下,StringBuilder 并发执行,而 StringBuffer 需要等待,所以 StringBuidler 快于 StringBuffer。