值类型与引用类型
众所周知,Java的数据类型分为两类,8种基本的数据类型(byte、short、int、long、float、double、char、boolean)和引用类型。基本数据类型通过对应的包装类与引用类型建立联系。与此同时,这两类数据的存储也不相同,基本数据类型,,也叫值类型,存放在栈内存当中,引用数据类型的变量在栈中仅仅存放引用类型变量的地址,其本身则存放在堆内存当中。
Java中==与equals的区别
==操作比较的是两个变量的值,对于引用类型表示的是两个变量在堆中存储的地址是否相同,即栈中的内容是否相同。
equals操作表示的两个变量是否是对同一个对象的引用,即堆中的内容是否相同,该方法继承自Java的根类Object。
String类的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方法,那么所有的类都自动继承Object类的equals方法
当String类调用equals方法时,当二者在栈内存的内容时,直接判断为true;否则,判断该类是否是String的实例,然后进一步比较字符串每个字符是否相等
测试用例
public static void main(String[] args) {
String str = new String("验证object类的tostrin方法");
String arr = new String("验证object类的tostrin方法");
System.out.println(str == arr);
System.out.println(str.equals(arr));
Person p1 = new Person(12,"peter");
Person p2 = new Person(12,"peter");
Person p3 = p1;
System.out.println(p1.equals(p2));
System.out.println(p1 == p2);
System.out.println(p3.equals(p1));
System.out.println(p3 == p1);
}
输出结果及分析:
false:引用的名称不同,直接判断为false
true:字符串的内容相同,判断为true
false:调用Object类的equals方法,引用名称不同,判断为false
false:根据“==”的规则直接判断为false
true:两对象是相同的引用
true:同上
总结:
1)对于==,如果作用于基本数据类型的变量,则直接比较其存储的 “值”是否相等;如果作用于引用类型的变量,则比较的是所指向的对象的地址
2)equals方法(equals方法不能作用于基本数据类型的变量),如果没有对equals方法进行重写,则比较的是引用类型的变量所指向的对象的地址;诸如String、Date等类对equals方法进行了重写的,比较的是所指向的对象的内容。
参考文献:Java核心技术,第九版
Java编程思想,第四版