在搞明白==和equals之前我们要搞明白一件事情,就是java中内存分类?下面是JDK1.7之前的内存结构图,JDK1.7之后只是把方法区变成了元空间,然后运行时常量池不在方法区中了,而是在堆中开辟了一块空间存放运行时常量。
java中内存可以简单的分类为栈内存和堆内存
栈内存:存放基本数据类型和引用变量,类似这种 int a = 1; String str = "aa";这种你创建的变量只涉及栈内存的空间,不会涉及堆内存。
堆内存:存放对象,类似这种 Person p = new Person(); 这种你创建的对象涉及两块内存,栈内存存放person实例的首地址,堆内存存放真正的对象内容。
01、==介绍
它的作用是判断两个对象的地址是不是相等。即,判断两个对象是不是同一个对象(基本数据类型==比较的是值,引用数据类型==比较的是内存地址)。
基本数据类型(包括他们的包装类型):byte,short,char,int,long,float,double,boolean。他们之间的比较,应用双等号(==),比较其地址,因为基本数据类型的值如果相同,那么其地址一定相同(这里不考虑数据的范围问题,下面有以Int类型举例范围对地址的影响),因为基本数据类型是存放在栈空间内的。
引用数据类型:当他们用(==)进行比较的时候,比较的是他们在内存中的存放地址(确切的说,是堆内存地址)。
public class IntegerSame {
@Test
public void test() {
int i = 100;//基本数据类型
int ii = 100;//基本数据类型
Integer j = 100;//引用类型
Integer jj = 100;//引用类型
Integer k = new Integer(100);//引用类型
Integer kk = new Integer(100);//引用类型
System.out.println("i的地址:" + System.identityHashCode(i));
System.out.println("ii的地址:" + System.identityHashCode(ii));
System.out.println("j的地址:" + System.identityHashCode(j));
System.out.println("jj的地址:" + System.identityHashCode(jj));
System.out.println("k的地址:" + System.identityHashCode(k));
System.out.println("kk的地址:" + System.identityHashCode(kk));
//基本类型相互比较其中的值,所以得出true
System.out.println("i == ii 结果:" + (i == ii));
//当int的引用类型Integer与基本类型进行比较的时候,包装类会先进行自动拆箱
//然后与基本类型进行值比较,所有得出true
System.out.println("i == j 结果:" + (i == j));
//同上,包装类先拆箱成基本类型,然后比较,得出true
System.out.println("i == k 结果:" + (i == k));
//都是引用类型,所有比较的是地址,因为j与jj的地址相同,所有true
System.out.println("j == jj 结果:" + (j == jj));
//都是引用类型,因为使用的是new,所以分配的地址不同,false
System.out.println("k == kk 结果:" + (k == kk));
}
}
输出结果:
答案:在-128~127的Integer值并且以Integer x = value;的方式赋值的参数,x会从包装类型自动拆箱成基本数据类型,以供重用!所以,j、jj的内存地址都是一样的!而new 会开辟不同的内存空间,所以k与kk的地址不同。
下面我们把100变成1000试试!
public class IntegerSame {
@Test
public void test() {
int i = 10000;//基本数据类型
int ii = 10000;//基本数据类型
Integer j = 10000;//引用类型
Integer jj = 10000;//引用类型
Integer k = new Integer(10000);//引用类型
Integer kk = new Integer(10000);//引用类型
System.out.println("i的地址:" + System.identityHashCode(i));
System.out.println("ii的地址:" + System.identityHashCode(ii));
System.out.println("j的地址:" + System.identityHashCode(j));
System.out.println("jj的地址:" + System.identityHashCode(jj));
System.out.println("k的地址:" + System.identityHashCode(k));
System.out.println("kk的地址:" + System.identityHashCode(kk));
//基本类型相互比较其中的值,所以得出true
System.out.println("i == ii 结果:" + (i == ii));
//当int的引用类型Integer与基本类型进行比较的时候,包装类会先进行自动拆箱
//然后与基本类型进行值比较,所有得出true
System.out.println("i == j 结果:" + (i == j));
//同上,包装类先拆箱成基本类型,然后比较,得出true
System.out.println("i == k 结果:" + (i == k));
System.out.println("j == jj 结果:" + (j == jj));
System.out.println("k == kk 结果:" + (k == kk));
}
}
注意:Java 基本类型的包装类的大部分都实现了常量池技术,即 Byte,Short,Integer,Long,Character,Boolean;前面 4 种包装类默认创建了数值[-128,127] 的相应类型的缓存数据,Character创建了数值在[0,127]范围的缓存数据,Boolean 直接返回True Or False。如果超出对应范围仍然会去创建新的对象。这里的Float和Double是没有使用常量池的缓存技术的。
类型 | 描述 |
---|---|
Boolean | 全部自动拆箱 |
Byte | 全部自动拆箱 |
Short | -128~127区间自动拆箱 |
Integer | -128~127区间自动拆箱 |
Long | -128~127区间自动拆箱 |
Float | 没有拆箱 |
Doulbe | 没有拆箱 |
Character | 0~127区间自动拆箱 |
02、equals()方法介绍
它的作用也是判断两个对象是否相等。但它一般有两种使用情况:
情况1:类没有覆盖 equals() 方法。则通过 equals() 比较该类的两个对象时,等价于通过“==”比较这两个对象。
情况2:类覆盖了 equals() 方法。一般,我们都覆盖 equals() 方法来比较两个对象的内容是否相等;若它们的内容相等,则返回 true (即,认为这两个对象相等)。
Boolean、Byte、Short、Integer、Long、Float、Doulbe、Character 8种基本类型的包装类都重写了 equals() 方法,所以比较的时候,如果内容相同,则返回 true,例如:
//因为内容相同,返回的都是true
System.out.println("j.equals(jj) 结果:" + (j.equals(jj)));
System.out.println("(k.equals(kk) 结果:" + (k.equals(kk)));
03、String类型的比较介绍
string是一个非常特殊的数据类型,它可以通过String x = value;的方式进行赋值,也可以通过String x = new String(value)方式进行赋值。
String x = value;方式赋予的参数,会放入常量池内存块区域中;
String x = new String(value)方式赋予的参数,会放入堆内存区域中,当成对象处理。
public class DemoEquals {
public static void main(String[] args) {
String a = new String("ab"); // a 为一个引用
String b = new String("ab"); // b为另一个引用,对象的内容一样
String aa = "ab"; // 放在常量池中
String bb = "ab"; // 从常量池中查找
System.out.println("a地址:" + System.identityHashCode(a));
System.out.println("b地址:" + System.identityHashCode(b));
System.out.println("aa地址:" + System.identityHashCode(aa));
System.out.println("bb地址:" + System.identityHashCode(bb));
//地址相同,所以返回true
if (aa == bb) {
System.out.println("aa==bb");
}
// 地址不同,非同一个对象,所以返回false
if (a == b) {
System.out.println("a==b");
}
//地址不同,但是内容相同,所以返回true
if (a.equals(b)) {
System.out.println("aEQb");
}
}
}
public boolean equals(Object anObject) {
if (this == anObject) {
return true;
}
if (anObject instanceof String) {
String anotherString = (String)anObject;
int n = value.length;
if (n == anotherString.value.length) {
char v1[] = value;
char v2[] = anotherString.value;
int i = 0;
while (n-- != 0) {
if (v1[i] != v2[i])
return false;
i++;
}
return true;
}
}
return false;
}
如果内容相同,则返回true!
04、总结
==是用来比较地址的,对于基本数据类型,其值相同,那么地址也一定相同(不考虑范围问题),对于字符串、数组、对象等,则无法使用==来比较值是否相同,因为这些类型的值相同,地址不一定相同。如果需要比较这些类型的值是否相同,一定要重写equals(),如果相同,返回true;否则,返回false! 注意:String类中默认已经重写好了equals()方法,其是通过把字符串转换成字符数组进行比较的。
全部评论