一、含义

在java中,==和equals都是比较是否相等的。

==:比较的是地址,也就是比较内存中所存储的数值是否相等。所以比较基本类型的数据或者两个引用变量(地址)是否相等。

equals:比较的是内容。比较的不是引用,而是因为里面存储的内容。比如,对象的比较,比较的不是对象的地址,而是对象里面的属性。

二、详解

int a = 1;
		int b = 1;
		System.out.println(a == b);

上面就是比较基本类型的,输出是正确,这是很简单的。但是如果是比较String呢?

String str1 = new String("hello");
		String str2 = new String("hello");
		System.out.println(str1 == str2);

输出就false。因为new创建的是两个对象的地址,这样str1和str2肯定是两个不同的对象地址,比较肯定是不一样的。

如果是下面这句呢?

System.out.println(str1.equals(str2));

那输出就是true。

这里展现的就是两者比较东西的不同。那接下来我们看看源码。

三、源码

equals是Object的一个比较方法,打开源码如下:

public boolean equals(Object obj) {
        return (this == obj);
    }

那这样不就是和==一样吗?是的,如果是直接调用Object里面的equals,那的确是用==比较,比较的也是内存里面的地址,那这样str1和str2应该是一样的。但是我们上面测试过了。结果是不一样,原因就是String类中对Object里面的equals方法做了重写。打开(快捷键F3)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;
    }

这个方法就是比较的是内容了。对源码有兴趣的自己看,不难。

所以为什么我们比较对象的时候,明明两个对象是不一样的地址,只不过一样的内容,用==就是false,用equsls就是正确的。

四、扩展


看看这个下面这个代码


String c = "hello";
		String d = "hello";
		String str1 = new String("hello");
		String str2 = new String("hello");
		System.out.println("c == d" + (c == d));
		System.out.println("c.equals(str1)" + c.equals(str1));
		System.out.println("str1 == str2" + (str1 == str2));
		System.out.println("str1.equals(str2)" + str1.equals(str2));

结果如下:


c == d 

true


c.equals(str1)

true


str1 == str2

false


str1.equals(str2)

true


按照上面说c和d是一个对象,那为什么回事true,其实这里不是这样的。这里并没有去new一个实例化对象出来。String c = “hello”;是在堆内存中新建一个char[]的字符数组,然后c作为一个引用变量指向这个数组地址。而d并不是直接新建一个hello,而是检查内存中是否存在hello,如果存在,这d直接指向这个地址,所以c==d才会返回true。

而String str1 = new String("hello");实际上是创建了两个引用。第一个是hello的堆内存地址,另外一个是str1的栈对象里面的地址。这样其实就是新建了两个对象,而str1是指向hello这个地址的。所以,比较str1和str2的时候,如果是比较地址,那肯定是不一样的,如果是比较内容,那就是一样的。