1. String
问题1:字符型常量(char)和字符串常量(String)的区别?
- 字符常量是单引号引起的一个字符;字符串常量是双引号引起的若干字符;
- 字符常量相当于一个整型值(ASCII值),可参与表达式运算;字符串常量代表一个地址值;
- 字符常量在Java中占2字节,字符串占若干字节;
八种基本类型:boolean(长度未定), char(2字节), byte(1字节), short(2字节), int(4字节), long(8字节), float(4字节), double(8字节)。
String不是基本类型。
问题2: String为什么是不可变?
String类使用final关键字修饰字符数组保存字符串,在Java 9之后,使用byte数组存储。
private final char[] value;
// Java 9之后
private final byte[] value;
问题3: StringBuffer 和 StringBuilder 的区别?
-
二者都继承自 AbstractStringBuilder类,在AbstractStringBuilder类中也使用字符数组保存字符串,但没有使用final关键字修饰,所有都是可变的;
-
二者构造器都调用父类构造方法;
-
String中对象不可变,可以理解为常量,线程安全;StringBuffer对方法加了同步锁,所有方法都加上了synchronized,是线程安全的;而StringBuilder是非线程安全的;
-
String类型每次改变都会生成新的对象;StringBuffer和StringBuilder针对对象本身进行操作。
问题4:字符串创建过程?
-
通过new关键字,JVM创建字符串对象但不存储于字符串池,可调用intern()方法
String a = new String("abc"); String b = "abc"; System.out.println(a == b); // false System.out.println(a.equals(b)); // true String a = new String("abc"); a = a.intern(); // 放进字符串常量池 String b = "abc"; System.out.println(a == b); // true System.out.println(a.equals(b)); // true
-
直接使用双引号,JVM会去字符串池寻找
问题5:==和equals()
==
的作用是判断两个对象的**地址**是不是相等;基本类型比较值,引用类型比较内存地址;
equals()
:也是判断对象是否相等;如果类没有覆盖equals()方法,等价于通过"=="比较;如果覆盖equals()方法,一般是比较内容是否相等。
String重写了equals()方法,所以是比较值
public boolean equals(Object anObject) {
if (this == anObject) { // 判断是否是同一个对象?
return true;
}
if (anObject instanceof String) { // 判断类型是否相同?
String aString = (String)anObject;
if (coder() == aString.coder()) { // 字符编码是否相同
return isLatin1() ? StringLatin1.equals(value, aString.value)
: StringUTF16.equals(value, aString.value); // 不同编码下的字符串
}
}
return false;
}
问题6: 为什么重写equals()是必须重写hashCode()?
hashCode()的作用是获取哈希码,一个int整数;作用是确定该对象在哈希表中的索引位置。以HashSet为例:当把对象加入HashSet时,先计算hashCode来确定插入位置;如果该位置没有对象,则假设不存在重复元素;否则,在调用equals()方法检查对象是否相同。
- 对象相等,hashCode一定相同;这里的相等是指,通过equals()比较两个对象时返回true。
- 对象相等,调用equals()都返回true;
- 对象有相同的hashCode,不一定相等;
hashCode()的默认行为是对堆上的对象产生独特值。如果没有重写hashCode(),则该class的两个对象无论如何都不会相等,即使两者指向相同的数据。
问题7、String不可变的好处?
- 在多线程中使用时安全的,不需要任何其他同步操作;
- String不可变,用来存储密码很安全;
- String不可变可以在Java运行时节省大量Java堆空间。因为不同的字符串变量可以引用池中相同的字符串。