还傻傻分不清楚equals和==的区别吗?看完就明白了

解决一个问题的最好方法就是发现一个问题产生的根源,即发现最本质的东西,再去解决它。

Java语言里面的equals()方法是交给开发者们自己去覆盖重写编写功能的,即让开发者去定义当满足什么条件时,两个Object是相等的。

equals方法是由Object提供的,允许子类进行重写。

equals()的原始代码实现如下:

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

而有一些博客说equals()是比较具体的内容是不是相同的,这个说法其实不太准确,以偏概全。这个说法的来源是很多教程都是以String类中的equals()为参照物进行举例,String类的equals()重写了Object对象的原始equals方法,只会比较具体的内容。

String类的equals()的具体实现代码如下:


public boolean equals(Object anObject) {
    if (this == anObject) {
        return true;
    }
    if (anObject instanceof String) {
        String anotherString = (String)anObject;
        int n = count;
        if (n == anotherString.count) {
            char v1[] = value;
            char v2[] = anotherString.value;
            int i = offset;
            int j = anotherString.offset;
            while (n-- != 0) {
                if (v1[i++] != v2[j++])
                    return false;
            }
            return true;
        }
    }
    return false;
}

而 == 是比较两个引用在内存中指向的是不是同一个对象,也就是在内存中的存储地址是不是一样的,两个对象的引用相同时(指向的是同一个对象),则 == 会返回true,否则返回false。

所以 == 是和Object类的原始equals()方法是等价的。

现在很多类都对equals进行了重写,这里用常见的String类来编写个测试Demo

public class TestString {

	public static void main(String[] args) {
  
		String string_1 = "codevald";
		String string_2 = "codevald";
		System.out.println(string_1 == string_2); //结果为true
	
	
		String string_3 = new String("codevald");
		String string_4 = "code" + new String("vald");
		System.out.println(string_3 == string_4); //结果为false
		System.out.println(string_3.equals(string_4)); //结果为true

	
	}


}

可以看到Demo中第一个输出语句结果为true

String string_1 = "codevald"会现在栈中创建一个对String类的引用变量string_1,然后会去字符串常量池中寻找有没有"codevald",因为是第一次创建没有,则会将"codevald"存放进字符串常量池。

String string_2 = "codevald" 又在栈中创建了String类的引用变量string_2,然后会去字符串常量池中寻找有没有"codevald",如果没有,则会创建并存放进常量池,并将string_2指向"codevald",如果已经有了,则直接令string_2指向"codevald"。上面的代码已经将字符串存储进了常量池,所以两个引用(string_1和string_2)指向同一个"codevald",返回true。

String string_3 = new String("codevald")会在堆中创建一个对象,并将在栈中存在的对象引用指向堆中的对象,而String string_4则是分别用了常量池中的字符串和存放对象的堆中的字符串,做 + 的时候会进行动态调用,最后创建完生成的对象仍然存放在堆中,而两个对象在堆中的地址是不一样的,所以 == 会返回false。

而String类重写了equals方法,比较的是内存中存放的数据的具体内容是不是相同的,这里存放的都是"codevald",所以会返回true。

总结

对于字符串变量来说,使用""和"equals"比较字符串时,其比较方法不同。""比较两个变量本身的值,即两个对象在内存中的首地址,"equals"比较字符串包含内容是否相同。

对于非字符串变量来说,如果没有对equals()进行重写的话,"==" 和 "equals"方法的作用是相同的,都是用来比较对象在堆内存中的首地址,即用来比较两个引用变量是否指向同一个对象。

posted @ 2021-02-14 11:36  codevald  阅读(425)  评论(0编辑  收藏  举报