一、String类
String 类的部分源码:
public final class String implements Serializable, Comparable<String>, CharSequence {
private final char[] value;
private int hash;
private static final long serialVersionUID = -6849794470754667710L;
private static final ObjectStreamField[] serialPersistentFields = new ObjectStreamField[0];
public static final Comparator<String> CASE_INSENSITIVE_ORDER = new String.CaseInsensitiveComparator();
public String concat(String str) {
int otherLen = str.length();
if (otherLen == 0) {
return this;
}
// 重新创建
char buf[] = new char[count + otherLen];
getChars(0, count, buf, 0);
str.getChars(0, otherLen, buf, count);
return new String(0, count + otherLen, buf);
}
....
}
从上面可以看出:
- String 类是flinal类,也就意味着 String不能被继承,
- String 类其实是通过char数组保存字符串
- String 对象一旦被创建就是固定不变的了,对String对象的任何改变都不影响到原对象,相关的任何改变都会生成新的对象
- String使用private final char value[]来实现字符串的存储,也就是说String对象创建之后,就不能再修改此对象中存储的字符串内容,就是因为如此,才说String类型是不可变的(immutable)。
二、字符串常量池
字符串的分配和其他对象分配一样,是需要消耗高昂的时间和空间的,而且字符串我们使用的非常多。JVM为了提高性能和减少内存的开销,在实例化字符串的时候进行了一些优化:使用字符串常量池
每当我们创建字符串常量时,JVM会首先检查字符串常量池,如果该字符串已经存在常量池中,那么就直接返回常量池中的实例引用。如果字符串不存在常量池中,就会实例化该字符串并且将其放到常量池中。由于String字符串的不可变性我们可以十分肯定常量池中一定不存在两个相同的字符串
下面创建 同一个 "chenssy"对象
String a = "chenssy";
String b = "chenssy";
String c = new String("chenssy");
创建字符串的方式:
- 使用 “” 引号创建字符串
- 使用new关键字创建字符串
①、单独使用 “” 引号创建的字符串都是常量,编译期就已经确定存储到String Pool中;
②、使用new String("")创建的对象会存储到heap中,是运行期新创建的; (new创建字符串时首先查看池中是否有相同值的字符串,如果有,则拷贝一份到堆中,然后返回堆中的地址;如果池中没有,则在堆中创建一份,然后返回堆中的地址(注意,此时不需要从堆中复制到池中,否则,将使得堆中的字符串永远是池中的子集,导致浪费池的空间)!)
③、使用只包含常量的字符串连接符如"aa" + "aa"创建的也是常量,编译期就能确定,已经确定存储到String Pool中;
④、使用包含变量的字符串连接符如"aa" + s1创建的对象是运行期才创建的,存储在heap中;
三、总结:
1、使用 “” 不一定创建String对象,使用new String,一定会创建对象
2、关于equals和==
- 对于==,如果作用于基本数据类型的变量(byte,short,char,int,long,float,double,boolean ),则直接比较其存储的"值"是否相等;如果作用于引用类型的变量(String),则比较的是所指向的对象的地址(即是否指向同一个对象)。
- equals方法是基类Object中的方法,因此对于所有的继承于Object的类都会有该方法。在Object类中,equals方法是用来比较两个对象的引用是否相等,即是否指向同一个对象。
- 对于equals方法,注意:equals方法不能作用于基本数据类型的变量。如果没有对equals方法进行重写,则比较的是引用类型的变量所指向的对象的地址;而String类对equals方法进行了重写,用来比较指向的字符串对象所存储的字符串是否相等。其他的一些类诸如Double,Date,Integer等,都对equals方法进行了重写用来比较指向的对象所存储的内容是否相等。