首先看一段代码:
String a = "Hello";
String b = "Hello";
String c = new String("Hello");
String d = new String("Hello");
System.out.println(a == b);
System.out.println(b == c);
System.out.println(c == d);
System.out.println(a.equals(b));
System.out.println(b.equals(c));
System.out.println(c.equals(d));
结果是true,false,false,true,true,true。这里就涉及到字符串池。“字符串池”,是Java为了提高内存利用率而采用的措施:当遇到String
a = "Hello";
这样的语句时,Java会先在字符串池中寻找是否已经存在"Hello"这个字符串,如果没有,则建立字符串"Hello"对象,然后变量 a
指向这个地址;然后遇到语句String b = "Hello",这时字符串池中已经有
"Hello"了,所以直接让变量b也指向这个地址,省去了重新分配的麻烦。而在Java中,操作符“==”对于两个基本型来说,是判断其内容是否相同,对于两个对象来说,则是判断其地址是否相同,所以a
== b返回 true。那么String c = new
String("Hello")又如何处理呢?如果是这种写法,则不会去访问字符串池,而是先为变量 c 开辟空间,然后将值写入空间。所以a
== c返回false,c ==
d同样返回false。至于String的equals方法,因为它比较的不是对象的地址,而是对象的值,所以都返回true就不奇怪了。
Java虚拟机有一个字符串池,保存着几乎所有的字符串对象。字符串表达式总是指向字符串池中的一个对象。使用new操作创建的字符串对象不指向字符串池中的对象但是可以使用intern方法使其指向字符串池中的对象(注:如果池中已经有相同的字符串--使用equals方法确定,则直接返回池中的字符串,否则先将字符串添加到池中,再返回)。池中两个相等的字符串如果使用“==”来比较将返回真。
当我们在利用'=='比较基本类型时,比较的是值
但是==运算符用于比较包装器对象时,是检测对象是否值向同一个存储区域。因此以下的比较通常不成立: Integer a = 1000; Integer b = 1000; if(a == b)...
但是Java实现却有可能让它成立:(将1000改为100) Integer a = 100; Integer b = 100;
if(a == b)... 这是为什么呢?
原来在此处涉及自动打包操作,将int类型自动打包成Integer类型。而自动打包规范要求boolean,byte,char<=127,介于-128~127之间的short和int被包装到固定的对象中,即同一内存区域。
为避免此类问题,建议使用equals方法比较对象是否相等,而不是用==运算符比较对象是否相同