java中的数据类型分为两类:原始数据类型(byte、short、char、int、long、float、double、boolean)和 复合数据类型
1.原始数据类型: 他们之间的比较,应用双等号(==),比较的是他们的值。
2.复合数据类型(类): 当他们用(==)进行比较的时候,比较的是他们在内存中的存放地址。
所以,除非是同一个new出来的对象,他们的比较后的结果为true,否则比较后结果为false。
JAVA当中所有的类都是继承于Object这个基类的,在Object中的基类中定义了一个equals的方法,这个方法的初始行为是比较对象的内存地址,但在一些类库当中这个方法被覆盖掉了,如String,Integer,Date在这些类当中equals有其自身的实现, 而不再是比较类在堆内存中的存放地址了,而是比较值。
Object中的equals方法代码:
public boolean equals(Object obj) {
return (this == obj);// 比较的是基于复合型数据类型的内存地址
} 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;
}
所以对于复合型数据类型之间的equals比较,在没有覆写该方法之前,他们之间的比较还是基于内存地址的比较。
下面举个String类型的例子:
1.
public class TestString {
public static void main(String[] args) {
String s1 = "Monday";
String s2 = "Monday";
if (s1 == s2)
{
System.out.println("s1 == s2");}
else{
System.out.println("s1 != s2");}
}
}
编译并运行程序,输出:s1 == s2说明:s1 与 s2 引用同一个 String 对象 -- "Monday"!
2.
public class TestString {
public static void main(String[] args) {
String s1 = "Monday";
String s2 = new String("Monday");
if (s1 == s2)
{System.out.println("s1 == s2");}
else
{System.out.println("s1 != s2");}
if (s1.equals(s2)) {System.out.println("s1 equals s2");}
else{
System.out.println("s1 not equals s2");}
}
}
我们将s2用new操作符创建
程序输出:
s1 != s2
s1 equals s2
说明:s1 s2分别引用了两个"Monday"String对象
原因解释:
字符串缓冲池:程序在运行的时候会创建一个字符串缓冲池。
当使用s2 = "Monday"这样的表达式创建字符串的时候,程序首先会在这个String缓冲池中寻找相同值的对象,
在第一个程序中,s1先被放到池中,所以在s2被创建的时候,程序找到相同值的s1,s2引用s2所引用的对象。
第二段程序中,使用了 new 操作符,他明白的告诉程序:"我要一个新的!不要旧的!"于是一个新的"Monday"Sting对象被创建在内存中。他们的值相同,但是位置不同。
3.
public class TestString {
public static void main(String[] args) {
String s1 = "Monday";
String s2 = new String("Monday");
s2 = s2.intern();
if (s1 == s2)
{System.out.println("s1 == s2");}
else
{System.out.println("s1 != s2");}
if (s1.equals(s2)) {System.out.println("s1 equals s2");}
else{
System.out.println("s1 not equals s2");}
}
} 这次加入:s2 = s2.intern();
程序输出:
s1 == s2
s1 equals s2
原因解释:java.lang.String的intern方法的返回值是字符串,但是它实际上是先检查该字符串是否在字符串缓冲池中存在,存在就返回该引用,不存在就会把该字符串加到字符串缓冲池中再返回该引用。