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方法的返回值是字符串,但是它实际上是先检查该字符串是否在字符串缓冲池中存在,存在就返回该引用,不存在就会把该字符串加到字符串缓冲池中再返回该引用。