在初学 Java 时,可能会经常碰到下面的代码:

  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 编程思想》一书中的原话:

  “关系操作符生成的是一个 boolean 结果,它们计算的是操作数的值之间的关系”。

  这句话看似简单,理解起来还是需要细细体会的。说的简单点,== 就是用来比较值是否相等。下面先看几个例子:

  1 public class Main { /** * @param args */
2 public static void main(String[] args) { // TODO Auto-generated method stub
3
4 int n=3; int m=3;
5
6 System.out.println(n==m);
7
8 String str = new String("hello");
9 String str1 = new String("hello");
10 String str2 = new String("hello");
11
12 System.out.println(str1==str2);
13
14 str1 = str;
15 str2 = str;
16 System.out.println(str1==str2);
17 }
18
19 }

 输出结果为 true false true

  n==m 结果为 true,这个很容易理解,变量 n 和变量 m 存储的值都为 3,肯定是相等的。而为什么 str1 和 str2 两次比较的结果不同?要理解这个其实只需要理解基本数据类型变量和非基本数据类型变量的区别。

  在 Java 中游 8 种基本数据类型:

  浮点型:float(4 byte), double(8 byte)

  整型: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?

  1 public class Main { /** * @param args */
2 public static void main(String[] args) { // TODO Auto-generated method stub
3 String str1 = new String("hello");
4 String str2 = new String("hello");
5
6 System.out.println(str1.equals(str2));
7 }
8 }

要知道究竟,可以看一下 String 类的 equals 方法的具体实现,同样在该路径下,String.java 为 String 类的实现。

  下面是 String 类中 equals 方法的具体实现:

  可以看出,String 类对 equals 方法进行了重写,用来比较指向的字符串对象所存储的字符串是否相等。

  其他的一些类诸如 Double,Date,Integer 等,都对 equals 方法进行了重写用来比较指向的对象所存储的内容是否相等。

  总结来说:

  1)对于 ==,如果作用于基本数据类型的变量,则直接比较其存储的 “值”是否相等;

    如果作用于引用类型的变量,则比较的是所指向的对象的地址

  2)对于 equals 方法,注意:equals 方法不能作用于基本数据类型的变量

    如果没有对 equals 方法进行重写,则比较的是引用类型的变量所指向的对象的地址;

    诸如 String、Date 等类对 equals 方法进行了重写的话,比较的是所指向的对象的内容。


本文出自于:tsg666

== 和 equal 区别的更多相关文章

  1. java中“==”和equal区别

    8个月以后就要正式找工作啦,我觉得现在是时候花时间好好深入研究一下以前比较混肴的知识.这就当作是自我成长的第一步! 对于String中的“equal方法”和“==”一直有点混肴,今天重新看了一下他们两 ...

  2. ruby中的== eql?和equal?区别

    原文 http://www.wellho.net/mouth/985_Equality-in-Ruby-eql-and-equal-.html Equality in Ruby - == eql? a ...

  3. java String final + equal == 区别

    java String 是不可改变的类型. String a = "hello2"; String d = "hello"; final String b = ...

  4. java String比较,“==”和“equal”区别

    public static void main(String[] args){ String str1 = new String("str"); String str2 = new ...

  5. java中 == 与 equal区别 转

    java中的数据类型,可分为两类:1.基本数据类型,也称原始数据类型.byte,short,char,int,long,float,double,boolean   他们之间的比较,应用双等号(==) ...

  6. JAVA面试之集合框架(三)

    21.ArrayList和Vector的区别 这两个类都实现了List接口(List接口继承了Collection接口),他们都是有序集合,即存储在这两个集合中的元素的位置都是有顺序的,相当于一种动态 ...

  7. 20道Java面试必考题

    系统整理了一下有关Java的面试题,包括基础篇,javaweb篇,框架篇,数据库篇,多线程篇,并发篇,算法篇等等,陆续更新中.其他方面如前端后端等等的面试题也在整理中,都会有的. 注:文末有福利!pd ...

  8. Java面试题—初级(3)

    21.ArrayList和Vector的区别 这两个类都实现了List接口(List接口继承了Collection接口),他们都是有序集合,即存储在这两个集合中的元素的位置都是有顺序的,相当于一种动态 ...

  9. [ Java面试题 ] 集合篇

    1.ArrayList和Vector的区别 这两个类都实现了List接口(List接口继承了Collection接口),他们都是有序集合,即存储在这两个集合中的元素的位置都是有顺序的,相当于一种动态的 ...

随机推荐

  1. 走进HTTP协议之二 基本HTTP机制

    本系列第一节,我们回顾了与HTTP协议有关的基本术语和概念,本文将分析HTTP协议的基本原理与机制 HTTP协议的用途 HTTP协议用于客户端与服务器之间的通信,在通信线路两端,必定一端是客户端,另一 ...

  2. NO.10: 在operator=中处理 "自我赋值"

    1.确保当对象自我赋值时operator=有良好的行为,其中的技术包括 "来源对象" 和 "目标对象" 的地址,精心周到的语句顺序,以及“ copy and s ...

  3. python类变量和实例变量的区别

    类变量:是为类服务的,类所有的实例都共享使用,在一个地方被改变,所有调用的地方变量值都改变.定义类时的写法为类名.变量名 实例变量:是在实例中生效的,每个实例变量的值都根据实例本身需求进行修改,不会影 ...

  4. vue2.0 之条件渲染

    条件渲染v-if.v-show <template> <div> <a v-if="isPartA">partA</a> <a ...

  5. HTTP协议和SOCKS5协议

    HTTP协议和SOCKS5协议 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 我们平时上网的时候基本上是离不开浏览器的,尤其是搜索资料的时候,那么这个浏览器是如何工作的呢?用的又是 ...

  6. JavaSE学习总结(五)——封装,继承,多态很简单

    java面向对象的三大特性是:封装.继承与多态,是面向对象编程的核心. 一.封装 简单说封装就是将同一类事物的特性与功能包装在一起,对外暴露调用的接口. 封装:封装也称信息隐藏,是指利用抽象数据类型把 ...

  7. string中substr,find函数使用

    2.string函数 find:某子串的起始位(0开始),函数的第二个参数使用代表从该位开始的后缀 substr:1) x开始的连续y位 2) x开始的后缀 #include<bits/stdc ...

  8. myeclispe 一直运行debug问题

    window->preferences->Myeclipse->Servers->Tomcat 然后找到你的相应的Tomcat服务器的版本,选中然后展开其下面的子菜单会发现有个 ...

  9. 计算机网络之互联网|因特网|万维网|HTTP|HTML之间的关系辨析

    本博文基于知乎"Web 是什么意思?"一问而引起.(本文均属于博主从知乎上自身所答搬运而至). 如无特殊声明,括号()内以分号分隔的名词均等效. 本文如无特殊引用声明,则所有内容版 ...

  10. luogu P4082 [USACO17DEC]Push a Box

    传送门 一个人推箱子,和之前的华容道中的棋子移动有异曲同工之妙,因为每次可以让人走到箱子的其他方向上,或者推一下箱子 所以状态可以设成\(f_{i,j,k}\),即箱子在\((i,j)\),人在\(k ...