String s1 = new String("string");
String s2 = new String("string");
System.out.println(s1 == s2);//false
System.out.println(s1.equals(s2));//true
首先看上面的代码,两个结果的输出不一样,让我们来探讨一下是如何产生这种差异的。
一:Java中==的理解
==是一个关系操作符,《Java编程思想》中这样描述:
“关系操作符生成的是一个Boolean结果,它们计算的是操作数的值之间的关系”,下面我们用代码解释
int n=3;
int m=3;
System.out.println(n==m);//true
String str = new String("hello");
String str1 = new String("hello");
String str2 = new String("hello");
System.out.println(str1==str2);//false
str1 = str;
str2 = str;
System.out.println(str1==str2);//true
既然是比较的操作数的值,但是相同值的对象为什么比较起来返回false?根本在于基本数据类型变量和非基本类型变量的区别
对于8中基本类型变量:
1B(byte,字节)= 8 bit(8 位);
-----整型:byte(1 B)、short(2 B)、int(4 B) 、long(8 B)
-----浮点型:float(4 B)、 double(8 B)
-----字符型:char(2 B)
------布尔型:boolean(JVM 没有明确规定其所占的空间大小,仅规定取值“true”“false”)
注意 char占两个字节 char是无符号的
--------java中内码(运行内存)中的char使用UTF16的方式编码,一个char占用两个字节,但是某些字符需要两个char来表示。所以,一个字符会占用2个或4个字节。
--------UTF8编码中,英文字符占用一个字节;绝大多数汉字占用三个字节,个别汉字占用四个字节
对于非基本数据类型的的变量,称为引用类型的变量,String str就是引用类型变量,引用类型变量的存储的是“其关联的对象在内存中的地址”,所以str1==str2返回false;
但是最后str1 和str2 都指向了str指向的对象,所以返回true。
二:equals
JDK源码:
public boolean equals(Object obj) {
return (this == obj);
}
equals 是Object的方法,所有继承Object的类都有该方法,通过源码,我们可以知道
equals比较的是两个对象在内存中的地址是否相等,即引用是否指向了同一个对象;
注意:String 、Double、Date、Integer等对equals方法进行了重写
重写后这些类的equals比较的***指向的对象所存储的内容是否相等***
所以下面的输出是true
String s1 = new String("string");
String s2 = new String("string");
System.out.println(s1.equals(s2));//true
总结:
==,如果作用于基本数据类型,比较的是数值,作用于引用类型的变量,比较所指向的对象的内存地址;
equals 作用于基本的引用类型的变量,比较的是引用是否指向了同一个对象,即对象的内存地址是否相等;
但是String Date Integer Double等类对equals进行了重写,比较的是对象的存储的内容