Java语言中equals()方法的使用可以说比较的频繁,但是如果轻视equals()方法,一些意想不到的错误就会产生。哈哈,说的有点严重了~

  先谈谈equals()方法的出身。equals()方法在java.lang.Object类中声明,由于Object类是Java所有类的基类,因此equals()方法在Java中无所不在,我也是惊呆了。先贴一

段java.lang.Object类的英文说明:  

  Class Object is the root of the class hierarchy. Every class has Object as a superclass. All objects, including arrays, implement the methods of this class.

  翻译一下:Object类是类继承树中的根类。每个类都把Object类作为其超类。所有的对象,包括数组,都实现了Object类所有的方法。

  

  java.lang.Object类中的equals()方法如何声明的呢?源代码是这么写的:

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

  稍微解释一下,这段代码就是说如果有 obj1.equals(obj2),那么返回值就是判断obj1和obj2堆地址是否相同(也就是说是否是同一块内存,即是否是同一个对象)。

  阶段性总结一下java.lang.Object类的equals()方法:

  1) equals()方法使用必须要有2个对象;

  2) equals()方法返回的是一个boolean值,如果是同一个对象则返回true,否则返回false;

  3) equals()方法本质还是使用了"=="双目运算符。("==":基本数据类型比较内容是否相同;引用数据类型比较对象hashCode是否相同,即引用指向的对象是否是同一个)。

    摘自java.lang.Object类的equals()方法说明:equal objects must have equal hash codes

  

  解释完Object类的equals()方法后,我试着用例题的形式来分析一下这些继承Object类的派生类equals()方法。一般来说Java核心类库中的类几乎都有equals()的重写,但是

自己写的类就没有重写咯。如果你没区别这两者就贸然使用equals()方法,那很可能还是在使用Object类的equals()方法。

类型一: Object类的派生类没有重写Object类的equals()方法

 class MyClass{
int x; MyClass(int i){
x=i;
} public static void main(String[] args){
MyClass m1=new MyClass(100);
MyClass m2=new MyClass(100);
if(m1.equals(m2)){
System.out.println("Both are equal");}
else{
System.out.println("Both are not equal");
}
}
}

   A代码编译时报出错误提示信息"equals()方法未定义"

   B.编译通过,抛出运行期异常.

   C.输出Both are equal.

   D.输出Both are not equal

  分析:做错这道题的原因很多,但是做对这题的原因只有一个,那就是真正理解了equals()方法的前因后果。本题中MyClass类隐继承了Object类(所有类的基类),但是并没

有重写Object类中的equals()方法。因此情况就是MyClass类和Object类的equals()方法是同一个,只不过MyClass类调用Object类的equals()方法而已。

 public class MethodOverrideVsOverload {
public boolean equals( MethodOverrideVsOverload other ) {
System.out.println("MethodOverrideVsOverload equals method reached" );
return true;
} public static void main(String[] args) {
Object o1 = new MethodOverrideVsOverload();
Object o2 = new MethodOverrideVsOverload(); MethodOverrideVsOverload o3 = new MethodOverrideVsOverload();
MethodOverrideVsOverload o4 = new MethodOverrideVsOverload(); if(o1.equals(o2)){
System.out.println("objects o1 and o2 are equal");
} if(o3.equals(o4)){
System.out.println("objects o3 and o4 are equal");
}
}
}
输出结果为:
  MethodOverrideVsOverload equals method reached
  objects o3 and o4 are equal

  分析:这道题不仅考察了"重写",也考察了"向上造型"现象。如果这道题做错,我感觉很可能是你认为下面这段代码是重写了java.lang.Object类的equals()方法。

 public boolean equals( MethodOverrideVsOverload other ) {
System.out.println("MethodOverrideVsOverload equals method reached" );
return true;
}

  其实不然,java.lang.Object类equals()方法的签名和这个方法的签名是不一样的,不一样在哪?一看就知道是参数列表嘛,两个字母差这么多呢~。所以啊,Object o1,o2

指向子类对象的情况并不是方法重写,只是发生了方法重载,o1和o2仍然会调用java.lang.Object类的equals()方法。不明白的建议温故一下方法重载(overload)&重写

(override)这两者。关于Java"向上造型"语法现象具体机制我会专门写一篇总结来解释的。

类型二:Object类的派生类重写了Object类的equals()方法

  其实在Java Core Class Library中几乎都重写了java.lang.Object类的equals()方法。下面仅仅举2个常见的重写equals()方法的类。

  (1) String类重写Object类的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;
}

  下面分析一下String类equals()方法的源代码: 

 if (this == anObject) {
return true;
} // 这段其实就是Object类的翻版,功能一模一样

  主要用于String类在constant pools(方法区常量池)的判断,如果是同一个对象那么就不用再判断两个对象是否内容相同了。写在开头是必须的,你说勒!

 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; // 一有内容不同就返回false
}
return true; // 字符串内容均相同则返回true
}
15 }

  如果传入的是String类的对象,那么就是两个对象的内容比较,如果一有内容不同就返回false,均相同则返回true。

  (2) Arrays类重写Object类的equals()方法

  Arrays类的equals()方法有些区别,就是只加了static修饰符,当然这仍然是方法重写(只要派生类访问权限大于等于超类,派生类返回值类型是超类的子类,并且方法签名相

同,那么就可以方法重写)。而且Array类的equals()方法还重载了蛮多的,下面就举其中的long型(第一个)的例子加以说明:

public static boolean equals(long[] a, long[] a2) {
if (a==a2)
return true; // 如果数组引用指向同一个数组对象,那么显然是相同,返回true
if (a==null || a2==null)
return false; // 两个比较的数组不能是null int length = a.length;
if (a2.length != length)
return false; // 如果两个数组不等长,显然也不可能相等 for (int i=0; i<length; i++)
if (a[i] != a2[i])
return false; // 如果两个数组等长,那么比较数组的元素,如果所有元素相同,那么这两个数组相同,反之不同
return true;
}

总结:

1) 对于equals()方法,无论如何都不能太轻视。在equals()内部其实也是有"=="在实现部分功能的。

2) 对于基本数据类型,"=="比较的变量的内容是否相同;对于引用数据类型,"=="比较的是对象的地址是否相同。

3) String类的equals()方法比较的是字符串是否内容相同。很多人误用,我猜可能就是这家伙惹的祸吧,嘻嘻~。

equals()方法详解的更多相关文章

  1. java数组、java.lang.String、java.util.Arrays、java.lang.Object的toString()方法和equals()方法详解

    public class Test { public static void main(String[] args) { int[] a = {1, 2, 4, 6}; int[] b = a; in ...

  2. toString()和equals()方法详解

    一:toString()方法 Object中toString方法 public String toString() { return getClass().getName() + "@&qu ...

  3. Java提高篇——equals()与hashCode()方法详解

    java.lang.Object类中有两个非常重要的方法: 1 2 public boolean equals(Object obj) public int hashCode() Object类是类继 ...

  4. java基础(十六)----- equals()与hashCode()方法详解 —— 面试必问

    本文将详解 equals()与hashCode()方法 概述 java.lang.Object类中有两个非常重要的方法: public boolean equals(Object obj) publi ...

  5. equals()与hashCode()方法详解

    java.lang.Object类中有两个非常重要的方法: 1 2 public boolean equals(Object obj) public int hashCode() Object类是类继 ...

  6. 【Java】equals()与hashCode()方法详解 (转)

    java.lang.Object类中有两个非常重要的方法: 1 2 public boolean equals(Object obj) public int hashCode() Object类是类继 ...

  7. Java中的equals和hashCode方法详解

    Java中的equals和hashCode方法详解  转自 https://www.cnblogs.com/crazylqy/category/655181.html 参考:http://blog.c ...

  8. 转:Java中的equals和hashCode方法详解

    转自:Java中的equals和hashCode方法详解 Java中的equals方法和hashCode方法是Object中的,所以每个对象都是有这两个方法的,有时候我们需要实现特定需求,可能要重写这 ...

  9. equals()方法和hashCode()方法详解

    equals()方法和hashCode()方法详解 1. Object类中equals()方法源代码如下所示: /** * Object类中的equals()方法 */ public boolean ...

随机推荐

  1. Javascript Tip(!!)

    var a:var b=!!a; a默认是undefined.!a是true,!!a则是false,所以b的值是false,而不再是undefined,也非其它值,主要是为后续判断提供便利. !!一般 ...

  2. static_cast” : 无法从“void (__thiscall CMainFrame::* )(NMTOOLBARA *,LRESULT *)”转换为“void (__thiscall CCmdTarget::* )(NMHDR *,LRESULT

    static_cast” : 无法从“void (__thiscall CMainFrame::* )(NMTOOLBARA *,LRESULT *)”转换为“void (__thiscall CCm ...

  3. c# list排序的实现方式

    实体类实现IComparable接口,而且必须实现CompareTo方法 实体类定义如下: class Info:IComparable { public int Id { get; set; } p ...

  4. jdbcTemplate简单使用

    package com.bizvane.spider.tools; import org.apache.commons.dbcp.BasicDataSource; import org.springf ...

  5. js上传文件到后台时序列化数据

    let fd = new FormData() // 定义传递的序列化对象,for (let i = 0; i < addArr.length; i++) { // addArr是选中文件的输入 ...

  6. CSS 定位 (Positioning)概述

    div.h1 或 p 元素常常被称为块级元素. 这意味着这些元素显示为一块内容,即“块框”. 与之相反,span 和 strong 等元素称为“行内元素”,这是因为它们的内容显示在行中,即“行内框”. ...

  7. hdu1520 Anniversary party

    Anniversary party HDU - 1520 题意:你要举行一个晚会,所有人的关系可以构成一棵树,要求上下级关系的人不能同时出现,每一个人都有一个rating值,要求使整个晚会的ratin ...

  8. codevs1105 过河

    1105 过河 2005年NOIP全国联赛提高组 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 题解 题目描述 Description 在河上有一座独木桥,一 ...

  9. NOIp2013 火柴排队【逆序对/思维】 By cellur925

    题目大意:给你两列数\(ai\)和\(bi\),你可以交换每列数中相邻的两个数,求一个最小交换次数使\(\sum_{i=1}^{n}(a_i-b_i)^2\) 最小. 最后满足条件的两个序列一定是各个 ...

  10. Vue实例生命周期+vueRoter

    Vue实例生命周期 vue生命周期之beforeCreate 实例创建之前除标签外,所有的vue需要的数据,事件都不存在 vue生命周期之created 实例创建之后,data和事件已经被解析到,el ...