在初学Java时,可能会常常碰到下面的代码:html
1 String str1 = new String("hello"); 2 String str2 = new String("hello"); 3 4 System.out.println(str1==str2); 5 System.out.println(str1.equals(str2));
为何第4行和第5行的输出结果不同?==和equals方法之间的区别是什么?若是在初学Java的时候这个问题不弄清楚,就会致使本身在之后编写代码时出现一些低级的错误。今天就来一块儿了解一下==和equals方法的区别之处。java
一.关系操做符“==”到底比较的是什么?编程
下面这个句话是摘自《Java编程思想》一书中的原话:post
“关系操做符生成的是一个boolean结果,它们计算的是操做数的值之间的关系”。spa
这句话看似简单,理解起来仍是须要细细体会的。说的简单点,==就是用来比较值是否相等。下面先看几个例子:指针
public class Main { /** * @param args */ public static void main(String[] args) { // TODO Auto-generated method stub int n=3; int m=3; System.out.println(n==m); String str = new String("hello"); String str1 = new String("hello"); String str2 = new String("hello"); System.out.println(str1==str2); str1 = str; str2 = str; System.out.println(str1==str2); } }
输出结果为 true false truecode
n==m结果为true,这个很容易理解,变量n和变量m存储的值都为3,确定是相等的。而为何str1和str2两次比较的结果不一样?要理解这个其实只须要理解基本数据类型变量和非基本数据类型变量的区别。htm
在Java中游8种基本数据类型:对象
浮点型:float(4 byte), double(8 byte)blog
整型:byte(1 byte), short(2 byte), int(4 byte) , long(8 byte)
字符型: char(2 byte)
布尔型: boolean(JVM规范没有明确规定其所占的空间大小,仅规定其只可以取字面值"true"和"false")
对于这8种基本数据类型的变量,变量直接存储的是“值”,所以在用关系操做符==来进行比较时,比较的就是 “值” 自己。要注意浮点型和整型都是有符号类型的,而char是无符号类型的(char类型取值范围为0~2^16-1).
也就是说好比:
int n=3;
int m=3;
变量n和变量m都是直接存储的"3"这个数值,因此用==比较的时候结果是true。
而对于非基本数据类型的变量,在一些书籍中称做为 引用类型的变量。好比上面的str1就是引用类型的变量,引用类型的变量存储的并非 “值”自己,而是于其关联的对象在内存中的地址。好比下面这行代码:
String str1;
这句话声明了一个引用类型的变量,此时它并无和任何对象关联。
而 经过new String("hello")来产生一个对象(也称做为类String的一个实例),并将这个对象和str1进行绑定:
str1= new String("hello");
那么str1指向了一个对象(不少地方也把str1称做为对象的引用),此时变量str1中存储的是它指向的对象在内存中的存储地址,并非“值”自己,也就是说并非直接存储的字符串"hello"。这里面的引用和C/C++中的指针很相似。
所以在用==对str1和str2进行第一次比较时,获得的结果是false。所以它们分别指向的是不一样的对象,也就是说它们实际存储的内存地址不一样。
而在第二次比较时,都让str1和str2指向了str指向的对象,那么获得的结果毫无疑问是true。
二.equals比较的又是什么?
equals方法是基类Object中的方法,所以对于全部的继承于Object的类都会有该方法。为了更直观地理解equals方法的做用,直接看Object类中equals方法的实现。
该类的源码路径为:C:\Program Files\Java\jdk1.6.0_14的src.zip 的java.lang路径下的Object.java(视我的jdk安装路径而定)。
下面是Object类中equals方法的实现:
很显然,在Object类中,equals方法是用来比较两个对象的引用是否相等,便是否指向同一个对象。
可是有些朋友又会有疑问了,为何下面一段代码的输出结果是true?
public class Main { /** * @param args */ public static void main(String[] args) { // TODO Auto-generated method stub String str1 = new String("hello"); String str2 = new String("hello"); System.out.println(str1.equals(str2)); } }
要知道究竟,能够看一下String类的equals方法的具体实现,一样在该路径下,String.java为String类的实现。
下面是String类中equals方法的具体实现:
能够看出,String类对equals方法进行了重写,用来比较指向的字符串对象所存储的字符串是否相等。
其余的一些类诸如Double,Date,Integer等,都对equals方法进行了重写用来比较指向的对象所存储的内容是否相等。
总结来讲:
1)对于==,若是做用于基本数据类型的变量,则直接比较其存储的 “值”是否相等;
若是做用于引用类型的变量,则比较的是所指向的对象的地址
2)对于equals方法,注意:equals方法不能做用于基本数据类型的变量
若是没有对equals方法进行重写,则比较的是引用类型的变量所指向的对象的地址;
诸如String、Date等类对equals方法进行了重写的话,比较的是所指向的对象的内容。