前提知识
java中有八种基本数据类型,基本数据类型是存储在栈内的,当它们之间通过==进行比较时,是比较值是否相等。除了基本数据类型,就是引用类型。比如常用的字符串:
String s=new String("abc");
String s1=s;
其中s是储存在栈内,"abc"是存储在堆中。另外记住只用通过new出来的对象,才会在堆中分配内存;当把s赋值给s1时,因为s1是引用类型,所以两者指向的内存是一样的,内存形式可以大意为下图所示:
基类——object
java中所有的类都是继承自object,基类中定义了equals()方法,可以看下源码:
通过源码可以很明确的看到,基类中的equals()其实就是说使用==运算符,那么这个==号又是在比较两个对象的什么?仔细看上图中的一句话,“which states that equal objects must have equal hash codes”;意思就是说如果两个对象有相同的哈希码,则equals()函数返回true,即==是在比较两个对象的哈希码;哈希码又是干嘛的,看一下图源码中的解释:
通过源码中的解释,可以得知不同的对象具有不同的哈希码,且这个是由该对象的在内存中的地址转换而来的;于是,我们可以通过哈希码来唯一的确定一个对象。
子类
当在java中定义一个子类的时候,该类都是Object的子类,如果直接继承基类中的equals()函数,则通过==和equals()函数判断两个对象,得出的结果是一样的。但是子类可以覆盖基类的方法,于是就有可能使得==和equals()得出不一样的结果;比如,String类型,其中==比较的是两个对象所引用的对象的内存地址是否一样,而equals()则判断两个对象所引用的内容是否一样。
String s1=new String("abc");
String s2=new String("abc");
String s3=s1;
System.out.println(s1==s2);
System.out.println(s1.equals(s2));
System.out.println(s1==s3);
System.out.println(s1.equals(s3));
输出:
下面,我们定义一个类,直接继承基类的equals()函数,然后比较它的对象:
A a1=new A();
A a2=new A();
System.out.println(s1==s2);
System.out.println(s1.equals(s2));
class A{
int a=9;
}
输出:
可以看到==和equals()的输出是一样的;
常量池
以下内容转载自link
String s=“abcd” 是一种非常特殊的形式,和new 有本质的区别。它是java中唯一不需要new 就可以产生对象的途径。以String s=“abcd”; 形式赋值在java中叫直接量,它是在常量池中而不是象new一样放在压缩堆中。 这种形式的字符串,在JVM内部发生字符串拘留,即当声明这样的一个字符串后,JVM会在常量池中先查找有有没有一个值为"abcd"的对象,如果有,就会把它赋给当前引用.即原来那个引用和现在这个引用指点向了同一对象, 如果没有,则在常量池中新创建一个"abcd",下一次如果有String s1 = “abcd”;又会将s1指向"abcd"这个对象,即以这形式声明的字符串,只要值相等,任何多个引用都指向同一对象.而String s = new String(“abcd”);和其它任何对象一样.每调用一次就产生一个对象,只要它们调用。也可以这么理解: String str = “hello”; 先在内存中找是不是有"hello"这个对象,如果有,就让str指向那个"hello".
如果内存里没有"hello",就创建一个新的对象保存"hello". String str=new String (“hello”) 就是不管内存里是不是已经有"hello"这个对象,都新建一个对象保存"hello"。
String s="abc";
String s1="abc";
System.out.println(s==s1);
System.out.println(s.equals(s1));
输出:
结论
(1) 对于基类对象,==判断它两的值是否相等。
(2) 如果子类直接继承基类的equals()函数,则==和equals()的含义是一样的,即指示两个对象所引用的对象是否是同一个,即它们的内存地址是否一样。
(3) 如果子类对基类的equals()函数进行了覆盖,那么要根据实际情况进行判断;比如常用的String类型,equals()判断的是两个对象的值是否一样。