首先理解一下Java中内存的管理方法,Java把内存划分为两种,堆内存和栈内存。

    堆内存是用来储存数组和对象的内存空间,栈内存是用来存放基本数据类型和对象的引用。栈内存中信息会在生命周期结束时自动释放,而堆内存中的信息则需要当栈中变量没有对该对象的引用,且垃圾回收站开始回收空间时才会释放。

    或许上面一段话比较容易理解,但是参考以下实例,或许会有更深的体会:

    1.两个字符串的比较。

    这种写法可能是我们平时最常见到的写法(为了方便,这里在每行代码前加了序号)

1    String s1  =   " China " ;
2  String s2  =   " China " ;
3  System.out.println( " s1 == s2 :  "   +  (s1  ==  s2));
4  System.out.println( " s1 equals s2 :  "   +  (s1.equals(s2)));

    根据我们的经验,第四行的代码应该没有问题,因为equals方法是调用Comparable接口中的compareTo方法,而String中的实现是对两个对象内容的比较,因此,第四行代码打印结果应该为true,那么第三行代码呢?第三行代码是对两个对象的比较,比较两个对象是否为同一个对象,也就是要比较s1和s2是否指向同一个String对象,或许多数朋友以为第三行代码的结果应该为false,但是意外的是,第三行代码同样为true。这是因为Java在执行的过程中,当遇到第一行代码时,会先到堆内存中是否有一个China的字符串对象,如果有,则会在栈中新建一个引用指向该对象,否则,就会在堆中新建一个China的字符串对象,然后在栈中新建一个堆该对象的引用,由此可知,其实在前两行代码中,只建立了一个字符串对象,那么第三行代码就容易理解了,其实s1和s2的比较只是指向同一个对象的两个引用的比较,所以结果同样是true。

    因此上面代码的最终运行结果为:

    s1 == s2 : true
    s1 equals s2 : true

2.对上面的代码稍做修改,如下:

1    String s1  =  new String( " China ") ;
2  String s2  =  new String( " China ") ;
3  System.out.println( " s1 == s2 :  "   +  (s1  ==  s2));
4  System.out.println( " s1 equals s2 :  "   +  (s1.equals(s2)));

    简单在定义两个字符串变量时添加一个new关键字,如大家所料,这次的运行结果是:

    s1 == s2 : false
    s1 equals s2 : true

    很明显,调用new关键字时,JVM不会去判断堆中是否已经存在了该对象,而是直接再新建一个对象,然后在栈中新建该对象的引用。