一 String的创建
七大数据类型中是没有String类型的!String是一个类,那么类的创建正常是需要构造函数创建,如使用new方法。String类的创建有三种方法:
String s = "abc";
String s = new String(); //可传参,String的构造函数非常多
String s = "abc"+"efg";
1) String s = "abc"方式创建
这种方式的创建比较特殊,不同于其他的类,没有使用构造函数的方式创建。
①首先会在字符串常量池中找是否存在"abc";若存在则s直接指向字符串常量池"abc"的地址
②若不存在,则在字符串常量池中创建"abc",然后s指向字符串常量池"abc"的地址
也就意味着这种方式创建几个不同的对象但值一样时,其实指向的是同一个对象。
String s1 = "abc";
String s2 = "abc";
System.out.println(s1 == s2); //true。可见是同一个对象
需要注意的是,这种方式创建字符串不会在堆内存中开辟空间,只会在字符串常量池中保留"abc"。当然jdk7之后字符串常量池也在堆中,之前是在方法区
2) String s = new String() 方式创建
这种方式不同于第一种,它的创建利用了构造函数,所以会在堆内存中开辟空间。
创建过程及其对象到底是存在堆中还是存在常量池中,有多种情况多种讨论,与jdk7之前还是之后版本有关,有点迷糊。。。。可参考:https://juejin.im/post/5c160420518825235a05301e
但只需要肯定s是指向堆内存中对象的。
String s1 = new String("aa");
String s2 = new String("aa");
System.out.println(s1==s2); //false
3)String s = “abc”+“efg”
会自动将两个字符串拼接起来后,然后跟第一种一样。
二 源码解析
String的成员变量有:
private final char value[];
private int hash; // Default to 0
可见String底层是由char数组实现,并且是final修饰。还有一个hash值。
String类重写了hashCode和equals方法。
hashCode的重写。其hashCode方法如下:
public int hashCode() {
int h = hash;
if (h == 0 && value.length > 0) {
char val[] = value;
for (int i = 0; i < value.length; i++) {
h = 31 * h + val[i];
}
hash = h;
}
return h;
}
可见String的hashCode值至于char数组value有关。也就是说两个不同的String对象,只要值一样,那hashCode值就一样。
equals的重写。
public boolean equals(Object anObject) {
if (this == anObject) {
return true;
}
if (anObject instanceof String) {
String anotherString = (String)anObject;
int n = value.length;
if (n == anotherString.value.length) {
char v1[] = value;
char v2[] = anotherString.value;
int i = 0;
while (n-- != 0) {
if (v1[i] != v2[i])
return false;
i++;
}
return true;
}
}
return false;
}
可见重写后的equals方法是比较两个对象的char[]数组是否相同。