为了讲清楚这个问题,我先给一段代码和运行结果,将问题跑出来,然后再解释问题。
代码示例
public class demo001 {
public static void main(String[] args) {
String str1 = new String("hello");
String str2 = "hello";
String str3 = "hello";
System.out.println(str1+"\n"+str2+"\n"+str3);//我们知道三个字符串都是相同字符串
System.out.println(str1 == str2);//输出false '=='判断两个变量的地址是否相同,这里两个变量的地址显然不同
System.out.println(str3 == str2);//输出true
System.out.println(str1.equals(str2));//输出true
System.out.println(str1.hashCode());//
System.out.println(str2.hashCode());
System.out.println(str3.hashCode());
}
}
运行结果:
先说String实例化方法
或许平时的你不会用到String的new构造实例的方法,但是实际上,String不是一个基本数据类型,而是一个类,是一个引用数据类型。它也有new构造的方法。
那么
String str1 = new String("hello");
String str2 = "hello";
这段代码的区别是什么呢?
通过new创建的字符串保存在堆中
而第二种方式创建的字符串放在常量池中
再看后面的String str3 = “hello”;
这段代码执行会发什么呢?
这段代码定义了一个str3变量,将“hello”赋给str3的过程是,先在常量池中寻找是否有这个字符串,如果没有,就在常量池中建立一个,如果有就让str3成为一个引用变量,引用值指向这个字符串。
在这里,我们的str3和str2显然都是指向常量池中的这个“hello”字符串
‘==’的含义
’==‘首先是一个运算符。其作用呢?
如果比较的是基本数据类型,那么比较的是其值是否相等,如果比较的是引用数据类型,则比较的是其地址是否相同。
那么str1和str2因为堆内存不一样,所以其存储地址不相同。
str2和str3因为都是引用同一个字符串,所以本质上只在内存中存储了一份,其地址相同、
String中equals函数的含义
equals用于比较两个对象的内容是否相等,equals不能用于基本数据类型的变量
这里我们三个变量str1、str2、str3的字符串内容都是一样的,所以相等
hashcode()函数
获取对象的哈希值,为什么这三个变量的哈希值一样呢?
这里str2、str3的哈希值相同是因为他们本质上只在内存中存储了一份,取了两个名字而已。故哈希值肯定相同。
为什么str1和str2哈希值相同呢?因为这两变量的值相同,所以再hashmap中他们的桶下标是相同的,也就是在同一个桶里面,但是因为hashmap的实现是通过一个桶(数组)加链表实现,这两个变量在同一个桶对应的哈希值相同,但是变量的key值不同,所在链表的上的位置不一样。
普通equals和String中的equals区别
普通equals是比较两个对象的地址是否相同,但在String中将equals进行了重写。使得其含义发生了变化:比较内容是否相同
如果没进行重写,那么我们上面的代码中的
System.out.println(str1.equals(str2));的输出应该是false,因为他们的地址不同