一起养成写作习惯!这是我参与「掘金日新计划 · 4 月更文挑战」的第13天,点击查看活动详情。
String、StringBuffer、StringBuilder 有什么区别?
上面这个问题几乎在所有Java初级面试中都会问到,面试者一般会从可变性、性能和线程安全三个方面回答。回答如下:
String 为字符串常量是不可变对象,StringBuffer 与 StringBuilder 为字符串变量是可变对象。String 每次修改相当于生成一个新对象,因此性能最低;StringBuffer 使用 synchronized 来保证线程安全,性能优于 String,但不如 StringBuilder。StringBuilder 为非线程安全类,StringBuffer 为线程安全类。
接下来我们开聊:
String基础概念
String 是标准的不可变类,一旦创建了String对象,它的值就无法改变了。对它的任何改动,就是创建了一个新对象,再把引用指向该对象。
String对象赋值后会在常量池中缓存,如果下次创建会判定常量池中是否有缓存,如果有就直接返回给创建者。如果需要对字符串做很多修改,那么应该选择使用 StringBuffer & StringBuilder 类。
创建字符串:
String str = "hello world";
//通过构造函数创建字符串:
String str1=new String("hello world");
字符串方法
1.计算字符串长度
String str = "hello world";
System.out.println(str.length());
2.连接字符串
三种方式:
String str = “lao” + “wang”;
String str = “lao”; str += “wang”;
String str = “lao”; String str2 = str + “wang”;
3.比较字符串
使用 String 和 new String 声明的对象是不同的,那有没有简单的方法,可以忽略它们的创建方式(有没有 new)而只对比它们的值是否相同呢?答案是肯定的,使用 equals() 方法可以实现,代码如下:
String s1 = "hi," + "lao" + "wang";
String s2 = "hi,";
s2 += "lao";
s2 += "wang";
String s3 = "hi,laowang";
System.out.println(s1.equals(s2)); // true
System.out.println(s1.equals(s3)); // true
System.out.println(s2.equals(s3)); // true
以上使用 equals 对比的结果都为 true。
如果要忽略字符串的大小写对比值可以使用 equalsIgnoreCase(),代码示例:
String s1 = "Hi,laowang";
String s2 = "hi,laowang";
System.out.println(s1.equals(s2)); // false
System.out.println(s1.equalsIgnoreCase(s2)); // true
s1.equals(s2) 执行的结果为:false,s1.equalsIgnoreCase(s2) 执行的结果为:true。
4.截取字符串
字符串的截取使用 substring() 方法,使用如下:
String str = "abcdef";
// 结果:cdef(从下标为2的开始截取到最后,包含开始下标)
System.out.println(str.substring(2));
// 结果:cd(从下标为2的开始截取到下标为4的,包含开始下标不包含结束下标)
System.out.println(str.substring(2,4));
5.JVM对字符串的优化
对于 String 的任何操作其实是创建了一个新对象,然后再把引用地址返回该对象,但 JVM 也会对 String 进行特殊处理,以此来提供程序的运行效率,比如以下代码:
String str = "hi," + "lao" + "wang";
经过 JVM 优化后的代码是这样的:
String str = "hi,laowang";
验证代码如下:
String str = "hi," + "lao" + "wang";
String str2 = "hi,laowang";
System.out.println(str == str2);
执行的结果:true。
这就说明 JVM 在某些情况下会特殊处理 String 类型。
6.String StringBuffer StringBuilder使用
字符串相关类型主要有这三种:String、StringBuffer、StringBuilder,其中 StringBuffer、StringBuilder 都是可以变的字符串类型,StringBuffer 在字符串拼接时使用 synchronized 来保障线程安全,因此在多线程字符串拼接中推荐使用StringBuffer。
StringBuffer 使用:
StringBuffer sf = new StringBuffer("lao");
// 添加字符串到尾部
sf.append("wang"); // 执行结果:laowang
// 插入字符串到到当前字符串下标的位置
sf.insert(0,"hi,"); // 执行结果:hi,laowang
// 修改字符中某个下标的值
sf.setCharAt(0,'H'); // 执行结果:Hi,laowang
StringBuilder 的使用方法和 StringBuffer 一样,它们都继承于 AbstractStringBuilder。
面试题(6道)
1."=="和equals的区别是什么?
“==”是运算符
- 如果比较的对象是基本数据类型,则比较的是其存储的值是否相等;
- 如果比较的是引用数据类型,则比较的是所指向对象的地址值是否相等(是否是同一个对象)
equals()是Object的方法,用来比较两个对象的内容是否相等。
- equals 方法不能用于比较基本数据类型,如果没有对 equals 方法进行重写,则相当于“==”,比较的是引用类型的变量所指向的对象的地址值。
- 一般情况下,类会重写equals方法用来比较两个对象的内容是否相等。比如String类中的equals()是被重写了,比较的是对象的值。
2.如何判断字符串为null?
str.equals(“”)
str.length()==0
3.String对象的intern()方法的作用?
intern() 方法用于查找常量池中是否存在该字符值,如果常量池中不存在则先在常量池中创建,如果已经存在则直接返回。
String s = "laowang";
String s2 = s.intern();
System.out.println(s == s2); // 返回 true
4.String 不可变性有哪些好处。
- 只有当字符串是不可变的,字符串常量池才能实现,字符串池的实现可以在运行时节约很多堆空间,因为不同的字符串变量都指向池中的同一个字符串;
- 可以避免一些安全漏洞,比如在 Socket 编程中,主机名和端口都是以字符串的形式传入,因为字符串是不可变的,所以它的值是不可改变的,否则黑客们可以钻到空子,改变字符串指向的对象的值,造成安全漏洞;
- 多线程安全,因为字符串是不可变的,所以同一个字符串实例可以被多个线程共享,保证了多线程的安全性;
- 适合做缓存的 key,因为字符串是不可变的,所以在它创建的时候哈希值就被缓存了,不需要重新计算速度更快,所以字符串很适合作缓存的中的 key。
5.String可以被继承吗?
不能被继承。因为 String 被声明为 final(最终类),所以不能被继承。
public final class String
implements java.io.Serializable, Comparable<String>, CharSequence {
//......
}
6.String是基础数据类型吗?
String 不是基础数据类型。基础数据类型有以下8中,分别为:boolean、byte、short、int、long、float、double、char。