1.关系操作符==
有些人对于关系操作符==起到什么作用还是有一些模糊。
在Java编程思想书中有以下这样一句话:
关系操作符生成的是一个boolean结果,它们计算的是操作数的值之间的关系。
从字面意思上好像没有那么容易理解,那么我们先来看看下面这个例子。
package test;
public class Test {
public static void main(String agrs[])
{
int a = 1;
int b = 1;
String s1 = new String("Hello");
String s2 = new String("Hello");
System.out.println(a==b);
System.out.println(s1==s2);
}
}
运行结果中的第一个true是a==b产生的结果,变量a和变量b都是整型变量,它们的初始值都为1,所以它们是相等的,这一点是大家所认同的。
那为什么第二个运行结果为false呢?s1和s2的值不都是Hello吗?想弄明白这一点我们要先了解Java中变量的类型。
Java中变量的类型分为基本数据类型变量和非基本数据类型变量。
Java中有以下8种基本数据类型:
浮点型:float(占4byte),double(占8byte);
整型:byte(占1byte),short(占2byte),int(占4byte)
long(占8byte);
字符型:char型(占2byte)
布尔型:Boolean(Java规范中未明确规定大小,但是变量只有true和false两种取值)
对于基本数据类型的变量而言,变量中存储的是具体类型的值,所以==操作符比较的就是两个变量具体的值,比如例子中1==1,结果肯定为true。
而s1和s2是非基本数据类型变量,它们称之为引用型变量(类似于C/C++中指针变量的概念),代码通过new关键字创建了两个String类对象并与s1、s2关联起来,此时s1和s2中存储的是对象在内存中的地址,并不是对象具体的值Hello,因为创建了两个String类的对象,它们在内存中的地址也不相同,s1和s2两个引用变量中的值也就不同,所以s1==s2产生的结果为false。
特殊情况
package test;
public class Test {
public static void main(String agrs[])
{
String s1 = "Hello";
String s2 = "Hello";
String s3 = new String("Hello");
System.out.println(s1==s2);
System.out.println(s1==s3);
}
}
既然上面说了s1和s2是引用类型变量,存储的是引用对象的内存地址,那为什么此时第一个输出为true呢?
这是因为代码中把Hello给了s1和s2引用,s1和s2中存储的都是Hello字符串在内存中的堆地址(类似于指向同一个对象),所以s1==s2结果为true。第二个输出为false是因为s3为新创建的String对象的引用地址与之前Hello字符串在内存中的地址不同,所以s1==s3产生的结果为false。
2.equals方法又是如何比较的
想知道equals方法的功能我们不如先了解一下这个方法的来源。equals方法来自与基类Object类,继承了Object的子类都有这个方法。为了更加直观的了解equals方法的功能,我们看看它在JDK中的源码。
public boolean equals(Object obj) {
return (this == obj);
}
可以通过源码看出equals方法内部实现也是==操作符产生结果,比较两个对象引用的地址是否相等(是否指向同一个对象)。
但是看看以下代码以及运行结果:
package test;
public class Test {
public static void main(String agrs[])
{
String s1 = new String("Hello");
String s2 = new String("Hello");
System.out.println(s1==s2);
System.out.println(s1.equals(s2));
}
}
第一个输出的false的原因上面已经说过了,但是第二个输出为什么是true呢?equals比较的不是两个引用变量是否指向同一个对象吗?可是s1和s2指向的不是同一个String对象为什么还会输出true?
让我们来看看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;
}
我们可以从源码中看到String类对equals方法进行了重写,对其功能进行了扩充,增加了比较字符串引用变量指向的对象存储的字符串是否相等。所以s1.equals(s2)返回的值为true。
对于一些基本数据类型的包装类来说,例如:Integer,Double,Long都对Object的equals方法进行了重写,其对比的是指向对象存储的值。
3.小结
==关系操作符
- 比较基本数据类型的变量时比较变量的中存储的值;
- 比较引用类型的变量时比较引用变量指向的对象的内存地址是否相等。
equals方法 - 基本数据类型没有equals方法;
- 如果没有对equals方法进行重写则比较两个引用变量指向对象的内存地址是否相等;
- 对于一些基本数据类型的包装类来说,例如:Integer,Double的equals方法比较的是引用变量指向对象中存储的值。