Java基础提升篇:equals()方法和“==”运算符

equals()

超类Object中有这个equals()方法,该方法主要用于比较两个对象是否相等。该方法的源码如下:

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

我们知道所有的对象都拥有标识(内存地址)和状态(数据),同时“==”比较两个对象的的内存地址,所以说使用Object的equals()方法是比较两个对象的内存地址是否相等,即若object1.equals(object2)为true,则表示equals1和equals2实际上是引用同一个对象。虽然有时候Object的equals()方法可以满足我们一些基本的要求,但是我们必须要清楚我们很大部分时间都是进行两个对象的比较,这个时候Object的equals()方法就不可以了,实际上JDK中,String、Math等封装类都对equals()方法进行了重写。下面是String的equals()方法:

 1public boolean equals(Object anObject) {
 2    if (this == anObject) {
 3        return true;
 4    }
 5    if (anObject instanceof String) {
 6        String anotherString = (String)anObject;
 7        int n = count;
 8        if (n == anotherString.count) {
 9        char v1[] = value;
10        char v2[] = anotherString.value;
11        int i = offset;
12        int j = anotherString.offset;
13        while (n-- != 0) {
14            if (v1[i++] != v2[j++])
15            return false;
16        }
17        return true;
18        }
19    }
20    return false;
21    }

对于这个代码段:if (v1[i++] != v2[j++])return false;我们可以非常清晰的看到String的equals()方法是进行内容比较,而不是引用比较。至于其他的封装类都差不多。

equals()方法和“==”运算符比较

首先笼统的来讲 “java中equals()方法和“==”运算符” 都是比较的地址,那为什么我们在使用中总会出现混淆的情况呢老是弄错呢,这是因为“重写equals()方法”和一些 “特殊情况”的存在。

有两种用法说明:

(1)对于字符串变量来说,使用“==”和“equals()”方法比较字符串时,其比较方法不同。

  • ==”比较两个变量本身的值,即两个对象在内存中的首地址。
  • equals()”比较字符串中所包含的内容是否相同。

比如:

1String s1,s2,s3 = "abc", s4 ="abc" ;
2s1 = new String("abc");
3s2 = new String("abc");

结果: s1==s2 是 false //两个变量的内存地址不一样,也就是说它们指向的对象不 一样,故不相等。 s1.equals(s2) 是 true //两个变量的所包含的内容是abc,故相等。

  • 注意(1):

如果:

1StringBuffer s1 = new StringBuffer("a");
2StringBuffer s2 = new StringBuffer("a");

结果: s1.equals(s2) //是false

解释: StringBuffer类中没有重新定义equals这个方法,因此这个方法就来自Object类,而Object类中的equals方法是用来比较“地址”的,所以等于false.

  • 注意(2):

对于s3和s4来说,有一点不一样要引起注意,由于s3和s4是两个字符串常量所生成的变量,其中所存放的内存地址是相等的,所以s3==s4是true(即使没有s3=s4这样一个赋值语句)

(2)对于非字符串变量来说,"=="和"equals"方法的作用是相同的都是用来比较其对象在堆内存的首地址,即用来比较两个引用变量是否指向同一个对象。

比如:

1class A
2{
3      A obj1   =    new A();
4      A obj2   = new A();
5}

结果: obj1==obj2是false obj1.equals(obj2)是false

但是如加上这样一句:obj1=obj2;

结果 : obj1==obj2 是true obj1.equals(obj2) 是true

总之

equals方法对于字符串来说是比较内容的,而对于非字符串来说是比较其指向的对象是否相同的。

== 比较符也是比较指向的对象是否相同的也就是对象在对内存中的的首地址。

String类中重新定义了equals这个方法,而且比较的是值,而不是地址。所以是true。

注意:

  • 如果是基本类型比较,那么只能用==来比较,不能用equals
 1public class TestEquals {
 2public static void main(String[] args) 
 3{
 4int a = 3;
 5int b = 4;
 6int c = 3;
 7System.out.println(a == b);//结果是false
 8System.out.println(a == c);//结果是true
 9System.out.println(a.equals(c));//错误,编译不能通过,equals方法
10//不能运用与基本类型的比较
11}
12}
  • 对于基本类型的包装类型,比如Boolean、Character、Byte、Shot、Integer、Long、Float、Double等的引用变量,==是比较地址的,而equals是比较内容的。
 1public class TestEquals {
 2public static void main(String[] args) 
 3{ Integer n1 = new Integer(30);
 4Integer n2 = new Integer(30);
 5Integer n3 = new Integer(31);
 6System.out.println(n1 == n2);//结果是false 两个不同的Integer对象,故其地址不同,
 7System.out.println(n1 == n3);//那么不管是new Integer(30)还是new Integer(31) 结果都显示false
 8System.out.println(n1.equals(n2));//结果是true 根据jdk文档中的说明,n1与n2指向的对象中的内容是相等的,都是30,故equals比较后结果是true
 9System.out.println(n1.equals(n3));//结果是false 因对象内容不一样,一个是30一个是31
10}
11}

这是Integer的实例,如果是其他的比如Double、Character、Float等也一样。

原文发布于微信公众号 - 好好学java(SIHAIloveJAVA)

原文发表时间:2018-05-15

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏专知

2018年SCI期刊最新影响因子排行,最高244,人工智能TPAMI9.455

2018年6月26日,最新的SCI影响因子正式发布,涵盖1万2千篇期刊。CA-Cancer J Clin 依然拔得头筹,其影响因子今年再创新高,达244.585...

1272
来自专栏搞前端的李蚊子

Html5模拟通讯录人员排序(sen.js)

// JavaScript Document  var PY_Json_Str = ""; var PY_Str_1 = ""; var PY_Str_...

5886
来自专栏Ryan Miao

ehcache报错

jfinal2.0+tomcat7+ehcache2.6.11+Linux Linux version 2.6.18-164.el5 (mockbuild@x8...

3729
来自专栏前端儿

Web 前端颜色值--字体--使用,整理整理

颜色值 CSS 颜色使用组合了红绿蓝颜色值 (RGB) 的十六进制 (hex) 表示法进行定义。对光源进行设置的最低值可以是 0(十六进制 00)。最高值是 2...

2272
来自专栏marsggbo

Udacity并行计算课程 CS344 编程作业答案

832
来自专栏MelonTeam专栏

Bitmap 源码阅读笔记

导语: Android 系统上的图片的处理,跟Bitmap 这个类脱不了关系,我们有必要去深入阅读里面的源码,以便在工作中能更好的处理Bitmap相关的问题...

2478
来自专栏小壮和前端

键盘事件 和键码

981
来自专栏linux驱动个人学习

高通Audio中ASOC的machine驱动

ASoC被分为Machine、Platform和Codec三大部分,其中的Machine驱动负责Platform和Codec之间的耦合以及部分和设备或板子特定的...

9714
来自专栏WOLFRAM

错觉艺术的巅峰,错觉图形大师M.C. Escher的不可能方块的可能模型

1333
来自专栏余生开发

echarts太阳分布图-饼图来回穿梭

var dom = document.getElementById("container");

1152

扫码关注云+社区