Java入门基础:基于源码深度剖析==和equals的区别,注意存在Java内存池的特殊情况

1.基本数据类型,也称原始数据类型。

byte,short,char,int,long,float,double,boolean
他们之间的比较,应用双等号(==),比较的是他们的值

// 基本数据类型比较值
int a = 1;
int b = 1;
float c = 1.0f;
		
System.out.println(a == b); //  true
System.out.println(a == c); //  true

2.类类型数据(类的对象)

当他们用(==)进行比较的时候,比较的是他们在内存中的存放地址,所以,除非是同一个new出来的对象,他们的比较后的结果为true,否则比较后结果为false。

String x = new String("Hello");
String y = new String("Hello");
String z = x;
		
System.out.println(x == y); // false,  比内存
System.out.println(z == x); // true, 比内存
System.out.println(x.equals(y)); // true, String类重写了equals比内容

JAVA当中所有的类都是继承于Object这个基类的,在Object中的基类中定义了一个equals的方法,这个方法的初始行为是比较对象的内存地 址,看JDK Object类的源码,底层实际使用的是==在比较:

但在一些类库当中这个方法被覆盖掉了,如String,Integer,Date在这些类当中equals有其自身的实现,而不再是比较类在堆内存中的存放地址了。例如 String类中被重写过的equals方法如下:

String类中的equals方法首先是使用了==号比较两个对象的内存是否一致,如果内存一致,则内容肯定也是一致的,然后判断两个字符的长度,如果长度一致,再分别比较每个字符是否一致。


所以, 对于复合数据类型之间进行equals比较,在没有覆写equals方法的情况下,他们之间的比较还是基于他们在内存中的存放位置的地址值的,因为Object的equals方法也是用双等号(==)进行比较的,所以比较后的结果跟双等号(==)的结果相同。

 3、注意Java内存池的特殊性

以下代码为true的原因是,m和n是直接用""号定义的字符串,String类存在内存优化机制,如果用""号定义,会优先在内存池查找是否已经存在内容相同的对象,存在则n直接引用该对象,不在创建新的内存,从而节省内存开销

String m = "Hello";
String n = "Hello";
System.out.println(m == n); // true

 

发布了36 篇原创文章 · 获赞 56 · 访问量 7004

猜你喜欢

转载自blog.csdn.net/nosprings/article/details/103973745