Java中equal和==区别
1.起因
在一段Java代码中,使用了两种实现方式。
//第一种命令行输入
int main (String[] args) {
if(args[0] == "-logdb"){
System.out.println("args==-logdb");
} else {
System.out.println("args!=-logdb");
}
}
//第二种
int main (String[] args) {
String[] args1 = {"-logdb"};
if(args1[0] == "-logdb"){
System.out.println("args==-logdb");
} else {
System.out.println("args!=-logdb");
}
}
结果发现,第一种显示的是args!=-logdb,而第二种显示的却是args!=-logdb。
2.原因
-
1.首先,在Java中,和equal是不一样的。对于基本类型,如int,char之类的,比较的是他们的值。而对于复合类型(类),当使用比较的就不是他们的值,而是他们的地址,的意思可以看成是参与比较的两个对象是不是同一个对象,即是否具有相同的地址。而equal的初始行为是比较地址,但在一些类中被重写覆盖,如String类中的equal方法:
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;
}
我们可以很明白地看出,在开始比较地址如果相同就返回true,后续则是比较String的内容。所以,当我们需要比较两个字符串是否相同的时候,最好使用equal来进行比较。
-
2.创建字符串对象原理
在创建一个字符串对象str = X时,jvm会拿着X去字符串缓存池中找到是否有内容相同的字符串,若有,则将str与其联系起来,若没有,则在池中创建。
使用new创建对象时,会在堆栈区中创建一个对象。
如果直接指定,如第二个例子,则会去字符串缓存池中检查是否存在字符串,不存在则创建,不会在堆栈区创建。
-
例子
String str1 = "abc";
System.out.println(str1 == "abc"); //true步骤:
栈中开辟一块空间存放引用str1;
字符串缓存池中开辟一块空间,存放String常量"abc";
引用str1指向池中字符串缓存池"abc";
-
str1所指代的地址即常量"abc"所在地址,输出为true;
String str2 = new String("abc");
System.out.println(str2 == "abc"); //false
步骤:
栈中开辟一块空间存放引用str2;
检查维护串池,若串池中有"abc",copy到堆中,若没有,在堆中创建,并添加到串池中
引用str2指向堆中的新建的String对象"abc";
-
str2所指代的对象地址为堆中地址,而常量"abc"地址在池中,输出为false;
String str3 = new String("abc");
System.out.println(str3 == str2); //false
步骤:
-
3.字符串缓存池
在程序运行过程中,会创建一个字符串缓存池。下面,让我们来看一看第一个例子。int main (String[] args) {
if(args[0] == "-logdb"){
System.out.println("args==-logdb");
} else {
System.out.println("args!=-logdb");
}
}
我们可以看到,在new出一个args字符串数组的时候,在字符串缓存池中,并没有存在着"-logdb"。
所以,这两个字符串进行比较时,比较的是他们的地址,一个位于堆中,一个位于缓存池中,地址当然是不同的,所以返回的是不等于。
int main (String[] args) {
String[] args1 = {"-logdb"};
if(args1[0] == "-logdb"){
System.out.println("args==-logdb");
} else {
System.out.println("args!=-logdb");
}
}
而在第二个例子中,我们先示例化了一个字符串数组,其中存在着"-logdb",所以,当后续比较时,程序先在String缓存池中寻找相同值的对象,即这两比较对象的地址其实是一样的,都是缓存池中的地址,所以返回等于。