在java中字符串的声明与初始化有如下几种方式:
1. String a = new String("abc");
2. String b = "abc"
这两种方式结果都是一样的,但是实现机制却不相同,接着往下看:
String a = "hello";
String b = "hello";
与
String c = new String("hello");
String d = new String ("hello");
请问
a==b true or false? c==d true or false?
a.equlas(b) true or false? c.equals(d) true or false?
String a= "hello";
String b="hello";
String c = new String("hello");
String d= new String ("hello");
System.out.println("a==b="+(a==b)+","+"c==d="+(c==d));
System.out.println("a.equals(b)="+a.equals(b) +" ,"+"c.equals(d)="+c.equals(d));
//result
a==b=true,c==d=false
a.equals(b)=true ,c.equals(d)=true
简而言之,java中在new时,总会产生一个新的对象,而新对象则需要分配新的内存,所以即使两个对象的内容是相同的,但是在内存中的首地址是不同的,即c,d中存储的数值是不相同的,所以表达式c==d将返回false。
而a==b=true 是因为在JVM中存在着一个字符串池,其中保存着很多String对象,并且可以被共享使用,a,b引用的是同一个常量池中的对象,当创建一个字符串常量时,会首先在字符串常量池中查找是否已经有相同的字符串被定义(其判断依据是String类的equals方法的返回值),若已经定义则直接获取对其的引用,此时不需要创建对象;若没有定义,则需要创建这个对象,然后将其加入到字符串池中,再将它的引用返回。(由于String是不可变类,一旦创建好了就不能被修改,因此String对象可以被共享且不会导致程序的混乱)。
equals是Object类提供的方法之一。每一个java类都继承Object类,所以每个对象都有equals这个方法。Obejct类中定义的equals方法是直接使用”==”运算符比较两个对象,所以在没有覆盖equlas方法的时候,equals与“==”运算符一样,比较的是引用。
因为a与b,c与d对象中的内容是相同的,所以表达式
a.equals(b)=true ,c.equals(d)=true
现在想想
a==c ? true or false?
a.equals(c) ? true or false?
看一张图:
a与c指的是同一个东西吗? a与c内容是一样的吗?答案显而易见,a==c为false a.equals(c)为true
String c = new String("hello");
可以人为地分为两个过程,第一个过程是新建对象的过程,new String("hello")
; 第二个过程是赋值的过程,即String c =
。
在进行第一个过程new String("hello")
时,等价于”hello”与new String()
两个操作。若在字符串池中不存在”hello”则会创建一个字符串常量”hello”,并将其添加到字符串池中;若存在,则不进行创建,然后new String()
会在堆中创建一个新的对象。
问题:
new String(“hello”)创建了几个对象?
答案:一个或者两个。如果常量池中有”hello”,那么只创建一个对象;如果没有,就会创建两个对象
参考:Java程序员面试笔试