一. == 与equals 的本质区别
1. == 是指比较 两个比较对象 在内存中放的值(一般都是在方法中进行逻辑处理,所以默认是放在栈中)。
如: int a = 1 ; int b =1; 因为int 是原始类型(byte , short, int , long,char , float,double, boolean等),所以 a 存放的 是 1 , b存放的 也是1,在进行 a== b 操作时,返回的是true。因为 原始类型没有实现 equals方法(或者可以理解为 原始类型不是对象,没有这样的方法),所以不能 进行 如 a.equals(b) 这样的操作。
又如: Integer o1 = new Integer(2); Integer o2 = new Integer(2); 因为 o1 和 o2 在这儿指向的是对象,所以在进行 o1==o2 操作时,比较的是 栈中存放的值(地址), 他们分别指向不同的对象(放在堆中),所以执行结果是false。
注: 这儿进行 System.out.println(o1); 时,得到的是数值 2,而不是o1存储的内容,是因为Integer 这个对象对toString()方法进行了重写,返回的是 o1的值。
2. equals 是对象的祖先Object类中的一个实现了的方法,他的源码
public boolean equals(Object paramObject)
{
return this == paramObject;
}
所以 也是 进行 == 比较的。即比较内存中的内容。
二. 关于String的 == 和equals
首先得说下String这个东西。java代码 被编译为class文件时,有一个常量池,代码中的非关键字(感觉这样的描述也不对,具体点就是 变量名,变量值,方法名,方法参数名,类名,包名 等等 ,太多列举不完)都以字符串的形式存储在里面。在class被classloader 加载进内存时,会在内存中 建一块 这个class的 运行时常量池 ,存放class文件的常量池(这是个人理解,还没去求证,但是应该大部分会加载进来)及其他更多的信息。
package com.one.study;
public class TestString {
private String s1 = "hello";
private static String s2 = "hello";
public static void main(String[] args) {
String hello = "hello";
String s3 = new String("hello");
}
public void hello()
{}
}
经过对编译出来的class进行查看,可以发现 s1 , s2 ,s3, hello 变量(自身) 的 值 和 方法名 hello 都指向 class的常量池中的同一个字符串 "hello" (被优化)。而在class被加载进内存之后,一个字符串也只会生成一个,所以他们的使用都是从这一个字符串中引用出来的。
所以s1 ,s2 ,hello 他们在内存中的值(地址) 都是指向 运行时常量池中的同一个字符串,所以在他们之间进行 == 操作的时候 ,结果是true。
而s3 因为是new 了一个String 对象,这个对象是存放在堆上的,所以s3中的值(地址) 是指向堆的,而不是指向运行时常量池的,但是在初始化时,传入的 "hello" 值,还是是从常量池中的"hello" copy 去的。
所以s1 和 s3 进行 == 操作时, 返回的是false。
下面说下String 的equals方法 。
因为String对equals方法进行了重写。贴一下源码
public boolean equals(Object paramObject)
{
if (this == paramObject) {
return true;
}
if ((paramObject instanceof String)) {
String str = (String)paramObject;
int i = this.count;
if (i == str.count) {
char[] arrayOfChar1 = this.value;
char[] arrayOfChar2 = str.value;
int j = this.offset;
int k = str.offset;
while (i-- != 0) {
if (arrayOfChar1[(j++)] != arrayOfChar2[(k++)])
return false;
}
return true;
}
}
return false;
}
他是 先 还是 对两个object 的内存值进行比较,如果相同,则直接返回true,如果不同,则判断比较对象是否是String,如果是再转换为String进行 内容上的比较。
所以 s1,s2,s3 之间进行 equals操作时,返回的都市true。