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

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);//true
System
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java字符串面试题通常涉及到StringStringBuilderStringBuffer类的使用,以下是一些常见的问题和答案: 1. StringStringBuilderStringBuffer有什么区别? - String是不可变的字符序列,每次对String进行修改都会创建一个新的String对象,因此在频繁修改字符串时效率较低。 - StringBuilderStringBuffer都是可变的字符序列,可以在原有对象上进行修改,因此在频繁修改字符串时效率较高。StringBuilder是非线程安全的,而StringBuffer是线程安全的。 2. String str="i"与 String str=new String("i")一样吗? - 不完全一样。String str="i"会先在常量池中查找是否存在值为"i"的字符串,如果存在则直接返回该字符串的引用;如果不存在则在常量池中创建一个新的字符串"i",并返回该字符串的引用。而String str=new String("i")则会在堆内存中创建一个新的字符串对象,并将该对象的引用赋值给str。 3. 如何将字符串转换为字节数组? - 可以使用getBytes()方法,例如: ```java String str = "hello"; byte[] bytes = str.getBytes(); ``` 4. 如何将字符串转换为字符数组? - 可以使用toCharArray()方法,例如: ```java String str = "hello"; char[] chars = str.toCharArray(); ``` 5. 如何将字符数组转换为字符串? - 可以使用String的valueOf()方法,例如: ```java char[] chars = {'h', 'e', 'l', 'l', 'o'}; String str = String.valueOf(chars); ``` 6. 如何将字符串转换为小写或大写? - 可以使用toLowerCase()和toUpperCase()方法,例如: ```java String str = "Hello"; String lowerStr = str.toLowerCase(); // 输出:hello String upperStr = str.toUpperCase(); // 输出:HELLO ``` 7. 如何将两个字符串拼接起来? - 可以使用concat()方法或"+"运算符,例如: ```java String str1 = "hello"; String str2 = "world"; String str3 = str1.concat(str2); // 输出:helloworld String str4 = str1 + str2; // 输出:helloworld ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值