首先理解一下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不会去判断堆中是否已经存在了该对象,而是直接再新建一个对象,然后在栈中新建该对象的引用。