值类型是存储在内存中的堆栈(以后简称栈),而引用类型的变量在栈中仅仅是存储引用类型变量的地址,而其本身则存储在堆中。
==操作比较的是两个变量的值是否相等,对于引用型变量表示的是两个变量在堆中存储的地址是否相同,即栈中的内容是否相同。
equals操作表示的两个变量是否是对同一个对象的引用,即堆中的内容是否相同。
==比较的是两个对象的地址,而equals比较的是两个对象的内容。
显然,当equals为true时,==不一定为true;
一、String中的equals和==
@Test
public void test1() {
String a = "qqq";
String b = "qqq"; //保存到字符串缓冲池里,当有相同值的对象,就引用该对象,否则重新创建字符串
System.out.println(a==b); //比较的是地址
System.out.println(a.equals(b)); //比较的是值
}
结果:
@Test
public void test2() {
String a = "money";
String b = new String("money");
System.out.println(a==b); //比较的是地址
System.out.println(a.equals(b)); //比较的是值
}
@Test
public void test4() {
String a = new String("money");
String b = new String("money");
System.out.println(a==b); //比较的是地址
System.out.println(a.equals(b)); //比较的是值
}
结果:
字符串缓冲池:
原来,程序在运行的时候会创建一个字符串缓冲池当使用 b = "qqq" 这样的表达是创建字符串的时候,程序首先会在这个String缓冲池中寻找相同值的对象,在第一个程序中,a先被放到了池中,所以在b被创建的时候,程序找到了具有相同值的 a
将b引用a所引用的对象"qqq",所以地址相同。
第二段程序中,使用了 new 操作符,他明白的告诉程序:"我要一个新的!不要旧的!"于是一个新的"money"Sting对象被创建在内存中。他们的值相同,但是位置不同。
@Test
public void test3() {
String a = "money";
String b = new String("money").intern();
System.out.println(a==b); //比较的是地址
System.out.println(a.equals(b)); //比较的是值
}
结果:
原来,(Java.lang.String的intern()方法"abc".intern()方法的返回值还是字符串"abc",表面上看起来好像这个方法没什么用处。但实际上,它做了个小动作:检查字符串池里是否存在"abc"这么一个字符串,如果存在,就返回池里的字符串;如果不存在,该方法会把"abc"添加到字符串池中,然后再返回它的引用。)
二、简单数据类型和封装类中的equals和==
Java为每一个简单数据类型提供了一个封装类,每个基本数据类型可以封装成对象类型。
除int(Integer)和char(Character),其余类型首字母大写即成封装类类型名。double (Double), float(Float),long(Long), short(Short),byte(Byte),boolean(Boolean).
以int和Integer为例说明
Java中int和Integer区别如下:
1.int是基本的数据类型,默认值可以为0;2.Integer是int的封装类,默认值为null;3.int和Integer都可以表示某一个数值;4.int和Integer不能够互用,因为他们两种不同的数据类型;
int a1=1;
int a2=1;
Integer b1 =new Integer (1);
Integer b2 =new Integer (1);
------------------------------
a1==b1 这个是成立的.表达式的值为 true ,Integer会自动拆箱。b1==b2 这个也是不成立的.表达式的值为 false,虽然是相同的数据类型,但是它们是两个对象,==比较的是2个对象的地址,它们的地址是不相等的,内容相等都是1;
b1.equals(b2)==true 这个是成立的,表达式的值为 true. 相同数据类型,两个对象,地址不同,内容相同, quals比较的是2个对象的内容,所以成立。
(a.equals(b),因为equals比较的是两个对象,所以a,b都不能为基本数据类型,否则会出编译错误。)同理,其它的封装类和基本类型也是这样的.
java中equals和==的区别
==比较的是2个对象的地址,而equals比较的是2个对象的内容。
@Test
public void test5() {
int a1 = 1;
int a2 = 1;
Integer c1 = 1;
Integer c2 = 1;
Integer b1 = new Integer(1);
int bb = b1;
Integer b2 = new Integer(1);
System.out.println(a1==a2);
System.out.println(b1==b2);
System.out.println(b1.equals(b2));
System.out.println(a1==b1); //Integer会自动拆箱,变成int
System.out.println(a1==bb);
System.out.println(a2==b2);
System.out.println(c1==c2);
System.out.println(c1.equals(c2));
System.out.println(c1==b1);
System.out.println(c1.equals(b1)); //.equals进行拆箱操作
System.out.println(c1==a1); //Integer会自动拆箱,变成int
}
结果: