Java equals 和 == 比较

原创
2016/04/06 14:41
阅读数 58

      java中的数据类型分为两类:原始数据类型(byte、short、char、int、long、float、double、boolean)和 复合数据类型
1.原始数据类型:  他们之间的比较,应用双等号(==),比较的是他们的值。
2.复合数据类型(类):  当他们用(==)进行比较的时候,比较的是他们在内存中的存放地址。
      所以,除非是同一个new出来的对象,他们的比较后的结果为true,否则比较后结果为false。
      JAVA当中所有的类都是继承于Object这个基类的,在Object中的基类中定义了一个equals的方法,这个方法的初始行为是比较对象的内存地址,但在一些类库当中这个方法被覆盖掉了,如String,Integer,Date在这些类当中equals有其自身的实现, 而不再是比较类在堆内存中的存放地址了,而是比较值。

      Object中的equals方法代码:
              public boolean equals(Object obj) {
                    return (this == obj);// 比较的是基于复合型数据类型的内存地址
              }

      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;
                }
      所以对于复合型数据类型之间的equals比较,在没有覆写该方法之前,他们之间的比较还是基于内存地址的比较。

      下面举个String类型的例子:
       1.
         public class TestString {
   public static void main(String[] args) {
   String s1 = "Monday";
   String s2 = "Monday";
   if (s1 == s2)
   {
   System.out.println("s1 == s2");}
   else{
   System.out.println("s1 != s2");}
  }
   }
 编译并运行程序,输出:s1 == s2说明:s1 与 s2 引用同一个 String 对象 -- "Monday"!
 2.
   public class TestString {
  public static void main(String[] args) {
   String s1 = "Monday";
   String s2 = new String("Monday");
   if (s1 == s2)
   {System.out.println("s1 == s2");}
   else
   {System.out.println("s1 != s2");}
   if (s1.equals(s2)) {System.out.println("s1 equals s2");}
   else{
   System.out.println("s1 not equals s2");}
  }
  }
  我们将s2用new操作符创建
 程序输出:
 s1 != s2
 s1 equals s2
 说明:s1 s2分别引用了两个"Monday"String对象

 原因解释:
     字符串缓冲池:程序在运行的时候会创建一个字符串缓冲池。
     当使用s2 = "Monday"这样的表达式创建字符串的时候,程序首先会在这个String缓冲池中寻找相同值的对象,
     在第一个程序中,s1先被放到池中,所以在s2被创建的时候,程序找到相同值的s1,s2引用s2所引用的对象。
     第二段程序中,使用了 new 操作符,他明白的告诉程序:"我要一个新的!不要旧的!"于是一个新的"Monday"Sting对象被创建在内存中。他们的值相同,但是位置不同。
 3.
   public class TestString {
  public static void main(String[] args) {
   String s1 = "Monday";
   String s2 = new String("Monday");
   s2 = s2.intern();
   if (s1 == s2)
   {System.out.println("s1 == s2");}
   else
   {System.out.println("s1 != s2");}
   if (s1.equals(s2)) {System.out.println("s1 equals s2");}
   else{
   System.out.println("s1 not equals s2");}
  }
  }

 这次加入:s2 = s2.intern();
 程序输出:
 s1 == s2
 s1 equals s2

 原因解释:java.lang.String的intern方法的返回值是字符串,但是它实际上是先检查该字符串是否在字符串缓冲池中存在,存在就返回该引用,不存在就会把该字符串加到字符串缓冲池中再返回该引用。

展开阅读全文
加载中

作者的其它热门文章

打赏
1
0 收藏
分享
打赏
0 评论
0 收藏
1
分享
返回顶部
顶部