String、StringBuffer、StringBuilder相关性质和面试题

本文详细介绍了Java中String、StringBuffer和StringBuilder的特性,包括它们的基本性质、不可变性、StringTable的工作原理,以及与基本类型、包装类之间的转换。分析了它们的创建方式、内存分配、线程安全性和性能差异,同时提供了相关的面试题及其解析,帮助理解这些概念的实际应用和内在机制。
摘要由CSDN通过智能技术生成

String、StringBuffer、StringBuilder相关性质和面试题

String基本性质

  • 可以字面量赋值,也可以通过new造对象赋值;
  • String声明为final类型,不可继承,且具有不可变性
  • String底层用final修饰的value[]数组存储,在jdk8.0之前用char[]数组,jdk9.0后用byte[]数组存储;(why?因为使用发现大部分String类型能够被一个字符存储下来,虽然汉字不行,但是大部分还是英文类型居多,这样用byte数组存储能明显节省空间);
  • String实现了Serializable接口和Comparable接口;

String的不可变性和StringTable

String s1 = "abc";
String s2 = "abc";
System.out.println(s1==s2);//true
  • 上述字符串的创建是以字面量的形式赋值的,而字面量在类的加载后是存储在字符串常量池StringTable中的。

在jdk6.0时,字符串常量池是放在方法区中(此时方法区称为永久代,占用虚拟机内存);jdk7.0时,字符串常量池被转移到堆中;jdk8.0后,字符串常量池仍然被保留在堆中(而永久代的概念被元空间取代,占用本地内存)。
问:为什么字符床常量池的位置需要从方法去到堆中区?
答: 因为如果StringTable放在方法区中,只有在full GC的时候才会回收空间,回收的效率并不高。实际中字符串会被大量的创建,由于回收效率低,更加容易导致方法区内存不足,而放在堆中,可以利用minor/major/full gc回收,回收效率更高。

  • StringTable的一个特点是不能存放两个相同内容的字符串的,且不可更改(只能重新创建)。在jdk7.0之前,StringTable底层是固定大小的Hashtable,长度为1009,所以不存在相同的两个字符串。jdk7.0之后,长度不固定,默认设置为60013。当字符串常量池中存储的字符串过多时,会导致哈希冲突严重,导致链表的长度变长,从而导致在调用String.intern()时降低性能。

结合上面两点可以得出,变量s1和s2指向的是字符串常量池中的同一个“abc”,因此判断结果为true

String s1 = "abc";
String s2 = "abc";
s1 = "hello";
//s1 += "def";拼接/replace函数同样是需要在StringTable重新创建
System.out.println(s1 == s2);//false
System.out.println(s1);//hello
System.out.println(s2);//abc

如果给s1重新赋值“hello”,会在字符串常量池新建一个“hello”,然后s1的指针指向“hello”,所以判断结果为false

  • +=拼接/replace()函数同样是需要在StringTable重新创建;

问:String对象的创建通过字面量和new的方式两者有什么不同?

String s1 = "abc";
String s2 = "abc";
String s3 = new String("abc");
String s4 = new String("abc");
System.out.println(s1 == s2);
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值