java中equals与==经常容易混淆,简单一点说就是equals比较的是值是否相等,是一种方法,==比较的两个对象在JVM中的地址,是一种操作符。

做了几个小实验比较结果。

实验一:

String str1="ab";
String str2="ab";
System.out.println(s1==s2);//true
System.out.println(str1.equals(str2));//true

这里的str1与str2都指向了常量池中的同一对象,所以System.out.println(s1==s2);返回为true,当然,str1与str2二者字符串的值也是相同的。

实验二:

 String str1="ab";
String str2="abc";
System.out.println(str1==str2);//false
System.out.println(str1.equals(str2));//false

这里的str1指向了字符串ab,初始时在常量池中并没有找到字符串abc,则开辟地址存储字符串abc,并将str2指向了abc字符串,所以str1与str2与并不是一个对象,他们的字符串值也不相同。

实验三:

 String str1="ab";
String str2="ab";
String str6=str2;
System.out.println(str1==str6);//true
System.out.println(str2==str6);//true
System.out.println(str1.equals(str6));//true
System.out.println(str2.equals(str6));//true

这里将str2的值赋值给str6,因为常量池中已经存在了ab这个字符串,所以str1、str2与str6共享了同样的对象,==与equals同样返回true

实验四:

 String str1= new String("abc");
String str2= "abc";
System.out.println(str1==str2);//false

创建了两个引用。创建了两个对象。两个引用分别指向不同的两个对象。
以上代码说明,只要是用new()来新建对象的,都会在堆中创建,而且其字符串是单独存值的,即使与栈中的数据相同,也不会与栈中的数据共享。

实验五:

 String str1="ab";
String str3=new String("ab");
String str4=new String("ab");
System.out.println(str3==str4);//false
System.out.println(str1.equals(str3));//true
System.out.println(str3.equals(str4));//true

因为str3与str4的字符串的值是相同的,所以str3.equals(str4)为true,但是由于str3与str4指向的不是同一个对象,所以str3==str4返回是false。同样因为str1与str3字符串的值是相同的,尽管str1与str3指向的不是同一个对象,str1.equals(str3)同样返回的是true

实验六:

 String str1="ab";
String str3=new String("ab");
String str4=new String("ab");
String str5=str3;
System.out.println(str1.equals(str3));//true
System.out.println(str3.equals(str4));//true
System.out.println(str3.equals(str5));//true
System.out.println(str4.equals(str5));//true
System.out.println(str3==str5);//true
System.out.println(str4==str5);//false
System.out.println(str1==str5);//false
System.out.println(str1.equals(str5));//true

这里将str3赋值给str5,所以str3与str5指向了同一对象,这样str3==str5返回为true值,str3.equals(str5)同样返回true值。由于str4与str5指向的不是同一对象,所以str4==str5返回false。但是也是因为字符串的值是相同的,所以str4.euqals(str5)返回为true值。

下面是以上实验所有的代码:

 public class Test
{
public static void main(String[] args)
{
String str1="ab";
String str2="abc";
String str6=str2;
String str3=new String("ab");
String str4=new String("ab");
String str5=str3;
System.out.println(str1==str6);//true
System.out.println(str2==str6);//true
System.out.println(str1.equals(str6));//true
System.out.println(str2.equals(str6));//true
System.out.println(str1==str2);//true
System.out.println(str1.equals(str2));//true
System.out.println(str3==str4);//false
System.out.println(str1.equals(str3));//true
System.out.println(str3.equals(str4));//true
System.out.println(str3.equals(str5));//true
System.out.println(str4.equals(str5));//true
System.out.println(str3==str5);//true
System.out.println(str4==str5);//false
System.out.println(str1==str5);//false
System.out.println(str1.equals(str5));//true
}
}

将String类换成Integer包装类,测试代码及结果如下:

 class TestNumber
{
public static void main(String[] args)
{
Integer str1=23;
Integer str2=23;
Integer str6=str2;
Integer str3=new Integer(23);
Integer str4=new Integer(23);
Integer str5=str3;
System.out.println(str1==str6);//true
System.out.println(str2==str6);//true
System.out.println(str1.equals(str6));//true
System.out.println(str2.equals(str6));//true
System.out.println(str1==str2);//true
System.out.println(str1.equals(str2));//true
System.out.println(str3==str4);//false
System.out.println(str1.equals(str3));//true
System.out.println(str3.equals(str4));//true
System.out.println(str3.equals(str5));//true
System.out.println(str4.equals(str5));//true
System.out.println(str3==str5);//true
System.out.println(str4==str5);//false
}
}

补充说明:

Integer类的equals源码

  public boolean equals(Object obj) {
if (obj instanceof Integer) {
return value == ((Integer)obj).intValue();
}
return false;
}

可见,Integer中的equals中的是直接取得数值进行比较

而String中的equals源码如下:

 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;
}

可见,String类中的源码与Integer中的有所不同,不同之处在于先采用了==方式进行比较,如果不是同一个对象再比较值,可见这个equals函数适用于存储在常量池中的字符串,例如

 String str1="ab";
String str2="ab";
System.out.println(str1.equals(str2));

str1与str2的比较equals方法中的

 if (this == anObject) {
return true;
}

进行比较得出的true

而下面的例子:

 String str3=new String("ab");
String str4=new String("ab");
System.out.println(str3.equals(str4));

str3与str4由于不是同一对象,所以if(this==anObject)不符合条件,所以进行下面字符串值的比较,最后返回true

下面的例子同理

String str5="abc";
String str6=new String("abc");
System.out.println(str5.equals(str6));

但这里还有一个陷阱,就是Integer类,可以参考这个类的源码,知道了Integer类缓存了-128到127,官方的解释是小的数字使用的频率比较高,超过这个范围的时候就会执行new Integer,可见如下代码

Integer a=1000;
Integer b=1000;
System.out.println(a==b);//false;

是正确的

但是一旦两个变量a与b被赋值为100,结果就是true

如下代码:

Integer a=new Integer(1000);
int b=1000;
Integer c=new Integer(1000);
Integer d=new Integer(1000);
System.out.println(a==b);//true
System.out.println(c==d);//false 

由于c与d是new Integer得到的结果,并没有使用缓存,所以c与d并不是同一个对象,输出的结果是false

但是a与b为什么是true呢?

因为Integer与int进行==比较的时候,java会把Integer自动拆箱,转为int类型,这样就a与b就指向了同一对象,所以会输出true

关于java中equals与==的区别的小实验的更多相关文章

  1. java中equals和==的区别 (转)

    java中equals和==的区别  值类型是存储在内存中的堆栈(以后简称栈),而引用类型的变量在栈中仅仅是存储引用类型变量的地址,而其本身则存储在堆中. ==操作比较的是两个变量的值是否相等,对于引 ...

  2. 【转】Java中equals和==的区别

    [转]Java中equals和==的区别 java中的数据类型,可分为两类: 1.基本数据类型,也称原始数据类型.byte,short,char,int,long,float,double,boole ...

  3. java中.equals和==的区别?

    Java中的equals是十分重要的,和= =要区别开来,孙卫琴的JAVA面向对象编程一书对这个做了阐述,现在小结其主要内容,而且要将 = =和 equals列为重要的对比概念来学习 1.声明格式   ...

  4. (转)Java中equals和==的区别

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

  5. Java 中 Equals和==的区别(转)

    另外一篇参考: https://blog.csdn.net/striverli/article/details/52997927 在谈论equals和==的区别前,我们先简单介绍一下JVM中内存分配的 ...

  6. Java中equals和==的区别?为什么重写equals方法后,一定要重写hashCode方法?

    首先明确一点,equals是方法,==是操作符. 1. 如果比较的是基本数据类型: 只讨论==,因为equals是不存在的,因为java中基本数据类型不能调用method的. 2. 如果比较的是引用类 ...

  7. java中equals和==的区别详解

    java中的数据类型,可分为两类: 1.基本数据类型. byte,short,char,int,long,float,double,boolean这八大原始数据类型他们之间的比较,使用“==”,比较的 ...

  8. java中equals和"=="的区别

    "=="号,它比较的是一个对象在内存中的地址值, 比如2个字符串对象String s1 = new String("str");String s2 = new ...

  9. Java中equals和“==””的区别,String特殊

    public class TestString { /* * java中的数据类型,可分为两类: * 1.基本数据类型,也称为原始数据类型.byte,short,char,int,long,float ...

随机推荐

  1. rpm包查看和解压(转)

    From:http://www.51testing.com/html/57/28557-205195.html 查看rpm包内容: rpm -qpl *.rpm 解压rpm包: rpm2cpio *. ...

  2. JavaWeb学习总结第三篇--走进JSP页面元素

    JavaWeb学习(三)—走进JSP页面元素 JSP:Java Server Pages,译为Java服务器页面.其脚本采用Java语言,继承了Java所有优点.JSP元素可以分为指令元素.脚本元素和 ...

  3. 一个中断服务子程序ISR

    请看下面的程序(一个中断服务子程序ISR),请指出这段代码的错误.)[中国台湾某著名CPU生产公司2005年面试题] 答案:(1)ISR不能返回一个值.如果你不懂这个,那么是不会被雇用的.(2)ISR ...

  4. 初学shell,为了练习sed,写了个简单的批量修改文件名的脚本,后来执行时发现系统竟然自带有一个rename命令,顺便也记下了

    1 #!/bin/bash   2 <<Comment   3 批量修改文件名的脚本   4 2015/10/24   5 webber   6 Comment   7 ARGS=2   ...

  5. 九度OJ 1153:括号匹配问题 (DP)

    时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:5193 解决:2248 题目描述: 在某个字符串(长度不超过100)中有左括号.右括号和大小写字母:规定(与常见的算数式子一样)任何一个左括 ...

  6. JAVA Socket基础(简单实现)

    学习Socket需要了解的几个概念: Socket 指的是互联网连接中的各个终结点.互联网连接是怎么创建的,通过IP地址加端口号,进行互通. A电脑(192.168.3.125:80)>> ...

  7. sap 图标查看

    showicon这个程序很不错,可以显示SAP里所有的ICON(图标). 用事务码SE38直接运行程序:showicon 即可. 显示列表之后,双击任何一个图标可以显示出每一个图标的详细信息.

  8. 打开蓝牙debug hci log

    Android4.2之前抓取hci log都是通过hcidump命令完成的,但是Android4.2 Bluetooth引入了Bluedroid,这是一个新的蓝牙协议栈.所以抓取hci log的方法也 ...

  9. API的理解和使用——键管理

    核心知识点: 1.键重命名:rename和renamenx,使用renamenx时newkey必须不存在,重命名后会使用del删除原来的键,如果值比较大也会会造成阻塞. 2.随机返回一个值:rando ...

  10. hadoop磁盘空间不均衡的解决办法

    hadoop集群在运行一段时间后,总是会出现某台机器的磁盘使用率特别高,有的使用率特别低,针对这种情况,hadoop提供了balancer工具调整磁盘负载 使用命令:start-balancer.sh ...