Equals与==

 

          String str1 = new String("hello");
	  String str2 = new String("hello");
	          
	  System.out.println(str1==str2); //false
          System.out.println(str1.equals(str2));//true

一.关系操作符“==”到底比较的是什么?

  下面这个句话是摘自《Java编程思想》一书中的原话:

  “关系操作符生成的是一个boolean结果,它们计算的是操作数的值之间的关系”。

  这句话看似简单,理解起来还是需要细细体会的。说的简单点,==就是用来比较值是否相等。下面先看几个例子:

 

public class Main {

    /**
     * @param args
     */
    public static void main(String[] args) {
        // TODO Auto-generated method stub
        
        int n=3;
        int m=3;
        
        System.out.println(n==m);
        
        String str = new String("hello");
        String str1 = new String("hello");
        String str2 = new String("hello");
        
        System.out.println(str1==str2);
        
        str1 = str;
        str2 = str;
        System.out.println(str1==str2);
    }

}

输出结果为 true false true

  n==m结果为true,这个很容易理解,变量n和变量m存储的值都为3,肯定是相等的。而为什么str1和str2两次比较的结果不同?要理解这个其实只需要理解基本数据类型变量和非基本数据类型变量的区别。

在Java中有8种基本数据类型:基本类型之间的比较,应该会将低精度类型自动转为高精度类型再比较

  浮点型:float(4 byte), double(8 byte)

  整型:byte(1 byte), short(2 byte), int(4 byte) , long(8 byte)

  字符型: char(2 byte)

  布尔型: boolean(1bit)

  对于这8种基本数据类型的变量,变量直接存储的是“值”,因此在用关系操作符==来进行比较时,比较的就是 “值” 本身。要注意浮点型和整型都是有符号类型的,而char是无符号类型的(char类型取值范围为0~2^16-1).

  也就是说比如:

  int n=3;

  int m=3; 

  变量n和变量m都是直接存储的"3"这个数值,所以用==比较的时候结果是true。

而对于非基本数据类型的变量,在一些书籍中称作为 引用类型的变量。比如上面的str1就是引用类型的变量,引用类型的变量存储的并不是 “值”本身,而是于其关联的对象在内存中的地址。比如下面这行代码:

  String str1;

  这句话声明了一个引用类型的变量,此时它并没有和任何对象关联。

  而 通过new String("hello")来产生一个对象(也称作为类String的一个实例),并将这个对象和str1进行绑定:

  str1= new String("hello");

  那么str1指向了一个对象(很多地方也把str1称作为对象的引用),此时变量str1中存储的是它指向的对象在内存中的存储地址,并不是“值”本身,也就是说并不是直接存储的字符串"hello"。这里面的引用和C/C++中的指针很类似。

  因此在用==对str1和str2进行第一次比较时,得到的结果是false。因此它们分别指向的是不同的对象,也就是说它们实际存储的内存地址不同。

  而在第二次比较时,都让str1和str2指向了str指向的对象,那么得到的结果毫无疑问是true。

二.equals比较的又是什么?

  equals方法是基类Object中的方法,因此对于所有的继承于Object的类都会有该方法。为了更直观地理解equals方法的作用,直接看Object类中equals方法的实现。

下面是Object类中equals方法的实现:

Equals与==_第1张图片

很显然,在Object类中,equals方法是用来比较两个对象的引用是否相等,即是否指向同一个对象。

为什么下面一段代码的输出结果是true?

public class Main {

    /**
     * @param args
     */
    public static void main(String[] args) {
        // TODO Auto-generated method stub
        
        String str1 = new String("hello");
        String str2 = new String("hello");
        
        System.out.println(str1.equals(str2));
    }
}

 

要知道究竟,可以看一下String类的equals方法的具体实现,同样在该路径下,String.java为String类的实现。

  下面是String类中equals方法的具体实现:

Equals与==_第2张图片

 可以看出,String类对equals方法进行了重写,用来比较指向的字符串对象所存储的字符串是否相等。

  其他的一些类诸如Double,Date,Integer等,都对equals方法进行了重写用来比较指向的对象所存储的内容是否相等。

  总结来说:

  1)对于==,如果作用于基本数据类型的变量,则直接比较其存储的 “值”是否相等;

    如果作用于引用类型的变量,则比较的是所指向的对象的地址

  2)对于equals方法,注意:equals方法不能作用于基本数据类型的变量

    如果没有对equals方法进行重写,则比较的是引用类型的变量所指向的对象的地址;

    诸如String、Date等类对equals方法进行了重写的话,比较的是所是否是同一类型的对象(不一定是同一对象),内容是否相等。

 

编写一个完美的equals方法的建议:


1)显示参数命名为otherObject,稍后将他转换成另一个叫做other的变量。*
2)检测this与otherObject是否引用同一个对象:
   if(this == otherObject)
   return true;
3)检测otherObject是否为null,如果为null,返回false
  if (otehrObject == null)
  return false;
4)比较this与otherObject是否属于同一个类。
   如果equals的语义在每个子类有所改变,就用getClass检测:
    if (getClass() != otherObject.getClass())
    return flase;
   如果所有子类都拥有统一的语义,就使用instanceof检测:
    if (!(otherObject instanceof ClassName))
    return flase;
5)将otherObject转换成相应的类类型变量:
  ClassName other=(ClassName)otherObject;
6)开始对所有需要比较的域进行比较。使用==比较基本数据类型,使用equals比较对象域。匹配返回true,否则返回false。
  return field1=other.field1
  && field2.equal(other.field2)
  &&....;

如果在子类中重新定义了equals,就要在其中包含调用super.equals(other)

对于数组类型的域,可以使用静态的Arrays。eauals方法检测相应的数组元素是否相等。

   static Boolean equals(type[] a,type[]b)
 如果两个数组长度相同,并且在对应的位置上数据元素均相同,将返回true数组元素类型可以是Object,int,long,short,char,byte,boolean,float,double. 

 

*  Equals与==_第3张图片

如果方法声明的显示参数类型不是otherObject,结果没有覆盖Object类的equals方法,而是定义了一个完全无关的方法。


将下列声明添加到Employee类中:
@Override

public boolean equals(Employee other)
就会看到一个错误报告,因为方法没有覆盖超类Object中的任何方法。

 

 

你可能感兴趣的:(Object,Java基础)