关系操作符”==”到底比较的是什么?
下面这个句话是摘自《Java编程思想》一书中的原话:
“关系操作符生成的是一个boolean结果,它们计算的是操作数的值之间的关系”。
简而言之,==就是用来比较值是否相等。
equals比较的是什么?
equals方法是基类Object中的方法,因此对于所有的继承于Object的类都会有该方法。为了更直观地理解equals方法的作用,直接看Object类中equals方法的实现。
public boolean equals(Object obj) {
return (this == obj);
}
很显然,在Object类中,equals方法是用来比较两个对象的引用(内存地址)是否相等,即是否指向同一个对象。
下面看看String类中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;
}
案例分析
public class Main {
public static void main(String[] args) {
// TODO Auto-generated method stub
int n=3;
int m=3;
System.out.println(n==m);
//↑true,
String str1 = "hello";
String str2 = "hello";
String str3 = new String("hello");
String str4 = new String("hello");
System.out.println(str1==str2);
/*
* JVM中存在着一个字符串池,其中保存着很多String对象;并且可以被共享使用。
* String有两个初始化的方式,一种是直接初始化,即声明时直接赋值的方式,另一个采用new关键字的方式来创建一个String类的实例对象。
* 两者有着截然不同的存取方式:
* 直接赋值是在常量池中创建一个字符串,而new关键字是每次都在堆区分配内存地址,创建新的对象,同时在栈区创建一个指向该变量的地址(引用)
* str1在字符串池中创建了一个了值为"hello"的被共享的对象,str2创建是在常量池中寻找是否存在值为"hello"的字符串,如果存在则指向该对象,
*不存在时创建一个新的对象。
*/
System.out.println(str1.equals(str3));
System.out.println(str1 == str3.intern());
/*
* String对象的实例调用intern方法后,可以让JVM检查常量池,
* 如果没有实例的value属性对应的字符串序列比如"hello"(注意是检查字符串序列而不是检查实例本身),
* 就将本实例放入常量池,
* 如果有当前实例的value属性对应的字符串序列"hello"在常量池中存在,
* 则返回常量池中"hello"对应的实例的引用而不是当前实例的引用
* */
System.out.println(str3==str4);
}
}
结果
true
true
true
true
false
总结来说:
1)对于==,如果作用于基本数据类型的变量,则直接比较其存储的 “值”是否相等;
如果作用于引用类型的变量,则比较的是所指向的对象的地址
2)对于equals方法,注意:equals方法不能作用于基本数据类型的变量
如果没有对equals方法进行重写,则比较的是引用类型的变量所指向的对象的地址;
诸如String、Date等类对equals方法进行了重写的话,比较的是所指向的对象的内容。